成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Jetty : Embedded Server 啟動流程 - 1

everfly / 2682人閱讀

前言

本文基于 Jetty 8.1.x 版本簡單介紹 Jetty Embedded Server 核心概念,線程模型,啟動流程。以下代碼片段摘自 Jetty 源代碼 中的 example-jetty-embedded 模塊的 OneServletContext.java

public class OneServletContext {
    public static void main(String[] args) throws Exception {
        Server server = new Server(8080);

        ServletContextHandler context = new ServletContextHandler(
        ServletContextHandler.SESSIONS);
        context.setContextPath("/");
        server.setHandler(context);

        ...
        
        // Serve some hello world servlets
        context.addServlet(new ServletHolder(new HelloServlet()),"/*");
        context.addServlet(new ServletHolder(
                new HelloServlet("Buongiorno Mondo")),"/it/*");
        context.addServlet(new ServletHolder(
                new HelloServlet("Bonjour le Monde")),"/fr/*");

        server.start();
        server.join();
    }
}

Embedded Jetty 使用起來很方便,少數(shù)幾行代碼就可以實現(xiàn)一個 http (Servlet)接口

創(chuàng)建 Server 對象,設(shè)置要監(jiān)聽的端口 8080

創(chuàng)建 ServletContextHandler 對象,ServletContextHandler "is-a" Handler(Request Handler),提供 ServletContext

將 ServletContextHandler 關(guān)聯(lián)到 Server(即 server.setHandler(context))

向 ServletContextHandler 中添加 Servlet,以 ServletHolder 的形式

啟動 Server

核心概念 LifeCycle

The lifecycle(生命周期) interface for generic components,聲明 start(啟動),stop(停止),isRunning,isStarted(查詢狀態(tài))等方法

AbstractLifeCycle

實現(xiàn) LifeCycle 接口的抽象類,提供生命周期管理默認實現(xiàn):例如通過 _state 屬性保存組件狀態(tài),提供 start,stop 默認實現(xiàn)

public final void start() throws Exception {
    // 線程安全
    synchronized (_lock) {
        try {
            // 狀態(tài)保護
            if (_state == _STARTED || _state == _STARTING) {
                return;
            }
            setStarting();
            // 將具體的 start 邏輯推遲到具體的子類實現(xiàn)
            doStart();
            setStarted();
        } catch (...) {
            ...
        }
    }
}
AggregateLifeCycle

An AggregateLifeCycle is an LifeCycle implementation for a collection of contained beans
AggregateLifeCycle 繼承自 AbstractLifeCycle,管理一組 Bean 的生命周期

public class AggregateLifeCycle extends AbstractLifeCycle implements
    Destroyable, Dumpable {
    private final List _beans = new CopyOnWriteArrayList<>();

    private boolean _started = false;

    @Override
    protected void doStart() throws Exception {
        for (Bean b: beans) {
            if (b._managed && b._bean instanceof LifeCycle) {
                LifeCycle l = (LefeCycle) b._bean;
                if (!l.isRunning()) {
                    // start managed bean
                    l.start();
                }
            }
        }
        _started =true;
        super.doStart();
    }
}
Server

類層次

AbstractLifeCycle
    AggregateLifeCycle
        AbstractHandler
            AbstractHandlerContainer
                HandlerWrapper
                    Server

Server 類是 Jetty 的《門面》類,包含:

Connector,接收客戶端請求

Handler,處理客戶端請求

ThreadPool,任務(wù)調(diào)度線程池

通過 Server 類提供的各種 set 屬性方法可以定制 Connector, ThreadPool,如果沒有特殊指定 Server 類會創(chuàng)建默認實現(xiàn),比如默認的 Connector 為 SelectChannelConnector,默認的 ThreadPool 為 QueuedThreadPool

Connector

(主要)類層次:

Connector
    AbstractConnector
        AbstractNIOConnector
            SelectChannelConnector

Connector 接口及其實現(xiàn)類用于 接收客戶端請求,(HTTP/SPDY)協(xié)議解析,最終調(diào)用 Server 中的(Request)Handler 處理請求,SelectChannelConnector 是 Server 默認使用的 Connector

public Server(int port) {
    setServer(this);

    Connector connector = new SelectChannelConnector();
    connector.setPort(port);
    setConnectors(new Connector[]{ connector });
}
Handler

Handler 或者叫作 Request Handler,用于處理客戶端請求

public interface Handler extends LifeCycle, Destroyable {
    ...
    void handle(String target, Request baseRequest, HttpServletRequest request,
            HttpServletResponse response);
    ...
}

handle 方法的簽名已經(jīng)很接近 Servlet service 方法,這里的 target 通常是請求的 uri,用于根據(jù) Servlet 配置規(guī)則找到具體的 Servlet 并調(diào)用其 service 方法

線程模型

Jetty Server 使用線程池來處理客戶端請求,ThreadPool 接口定義了線程池基本操作,Server 默認使用 QueuedThreadPool 類,如果需要,可以通過 Server.setThreadPool 方法手動設(shè)置線程池
注:Jetty 9.x 線程模型比較復(fù)雜,Jetty 8.x 比較簡單,代碼好讀一些~

QueuedThreadPool

QueuedThreadPool 類層次結(jié)構(gòu)

ThreadPool
    SizedThreadPool
        QueuedThreadPool

線程池核心問題:

線程管理:包括創(chuàng)建,銷毀等

任務(wù)分派:推或拉模型

線程管理

通過最小線程個數(shù),最大線程個數(shù),線程最大空閑時間 來動態(tài)創(chuàng)建,銷毀線程

線程池初始化
// QueuedThreadPool.java
public class QueuedThreadPool {
    // 默認最大線程數(shù)
    private int _maxThreads = 254;
    // 默認最小線程數(shù)
    private int _minThreads = 8;
    
    ...

    @Override
    protected void doStart() throws Exception {
        ...
        int threads = _threadsStarted.get();
        // 啟動 _minThreads 個線程
        while (isRunning() && threads < _minThreads) {
            startThread(threads);
            threads = _threadStarted.get();
        }
    }
}
線程銷毀

當(dāng)線程空閑時間超過設(shè)定的閾值 _maxIdleTimeMs 而且線程池的線程個數(shù)高于 _minThreads 時,線程將從 Runnable run 方法中退出

# QueuedThreadPool.java
private Runnable _runnable = new Runnable() {
    public void run() {
        boolean shrink = false;
        ...
        try {
            Runnable job = _jobs.poll()
            while (isRunning()) {
                // job loop:從隊列中拉取 job 并執(zhí)行直到隊列為空
                while (job != null && isRunning()) {
                    runJob(job);
                    job = _jobs.poll();
                }

                // idle loop
                try {
                    _threadsIdle.incrementAndGet();
                    while (isRunning() && job == null) {
                        // 如果 _maxIdleTimeMs < 0 就進入阻塞模式,直到成功拉取到 job
                        if (_maxIdleTimeMs <= 0) {
                            job = _jobs.take();
                        } else {
                            final int size = _threadsStarted.get();
                            // 如果線程個數(shù)少于 _minThreads 那么不需要回收線程
                            if (size > _minThreads) {
                                long last = _lastShrink.get();
                                long now = System.currentTimeMillis();
                                if (last == 0 || (now - last) > _maxIdleTimeMs) {
                                    shrink = _lastShrink.compareAndSet(last, now) &&
                                        _threadsStarted.compareAndSet(size, size - 1);
                                    if (shringk) {
                                        // 退出 while 循環(huán),即終止線程
                                        return;
                                    }
                                }
                            }
                            // 帶 timeout 的 poll
                            job = idleJobPoll();
                        }
                    }
                }
            }
        } catch (...) {
        } finally {
        }
    }
}
任務(wù)分派

使用阻塞隊列,工作線程從阻塞隊列中主動拉取(poll)任務(wù)(job),QueuedThreadPool 默認使用 ArrayBlockingQueue,如果需要可以通過構(gòu)造方法手動指定阻塞隊列的具體實現(xiàn)

public class QueuedThreadPool {
    public QueuedThreadPool(BlockingQueue jobQ) {
        this();
        _jobs = jobQ;
        _jobs.clear();
    }

    public boolean dispatch(Runnable job) {
        if (isRunning()) {
            final int jobQ = _jobs.size();
            final int idle = getIdleThreads();
            // 將 job 放入隊列中
            if (_jobs.offer(job)) {
                // 如果當(dāng)前沒有 idle 的線程,或者 隊列中堆積的 job 比 idle 線程多那么嘗試啟動新的線程
                if (idle == 0 || jobQ > idle) {
                    int threads = _threadsStarted.get();
                    if (threads < _maxThreads) {
                        startThread(threads);
                    }
                }
                return true;
            }
        }
        return false;
    }
}
啟動流程

Server 的啟動開始于 start 方法的調(diào)用,如上文所述,Server 類繼承自 AggregateServer,父類的 start 方法最終調(diào)用 Server 的 doStart 方法

Server start
@Override
protected void doStart() throws Exception {
    ...
    // 創(chuàng)建默認的 ThreadPool
    if (_threadPool == null) {
        setThreadPool(new QueuedThreadPool())
    }
    try {
        super.doStart();
    } catch(Throwable e) {...}
    // 啟動 Connector,接收,處理請求
    if (_connectors != null && mex.size == 0) {
        for (int i = 0; i < _connectors.length; i++) {
            try {
                _connectors[i].start();
            } catch (Throwable e) {
                mex.add(e);
            }
        }
    }
    ...
}

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/71172.html

相關(guān)文章

  • Jetty : Embedded Server 啟動流程 - 2

    摘要:前言本文接著啟動流程往下講,上回說到調(diào)用方法,開始接收和處理請求,默認使用的子類,所以我們重點來看看的具體實現(xiàn)類層次,我們上文提到過,這里有兩個新面孔抽象實現(xiàn)類上文講過方法最終會調(diào)用方法,使用模板方法模式在方法中實現(xiàn)了的基本流程子類方法, 前言 本文接著 Jetty : Embedded Server 啟動流程 - 1往下講,上回說到 Server.start 調(diào)用 Connector...

    phpmatt 評論0 收藏0
  • Spring MVC實現(xiàn)Spring Security,Spring Stomp websocket

    摘要:使用框架各個組件實現(xiàn)一個在線聊天網(wǎng)頁,當(dāng)有用戶連接,服務(wù)器監(jiān)聽到用戶連接會使用推送最新用戶列表,有用戶斷開刷新在線列表,實時推送用戶聊天信息。根據(jù)請求頭是否等于判斷是否是。 使用Spring框架各個組件實現(xiàn)一個在線聊天網(wǎng)頁,當(dāng)有用戶連接WebSocket,服務(wù)器監(jiān)聽到用戶連接會使用Stomp推送最新用戶列表,有用戶斷開刷新在線列表,實時推送用戶聊天信息。引入Jetty服務(wù)器,直接嵌入整...

    shuibo 評論0 收藏0
  • 《Spring Boot 編程思想 - 核心篇》勘誤匯總

    摘要:如果您在閱讀編程思想核心篇或示例練習(xí)的過程中發(fā)現(xiàn)了其中錯誤或提出建議,請將內(nèi)容提交至勘誤匯,小馬哥將勘誤或建議內(nèi)容匯總到此,修正后的內(nèi)容將在后續(xù)的書籍發(fā)行中體現(xiàn),并刊登勘誤貢獻者。筆者水平有限,行文的過程中錯誤無法避免,為此深表歉意。 如果您在閱讀《Spring Boot 編程思想 - 核心篇》或示例練習(xí)的過程中發(fā)現(xiàn)了其中錯誤或提出建議,請將內(nèi)容提交至【勘誤匯】,小馬哥將勘誤或建議內(nèi)容...

    trilever 評論0 收藏0
  • Spring Boot 2 快速教程:WebFlux 快速入門(二)

    摘要:響應(yīng)式編程是基于異步和事件驅(qū)動的非阻塞程序,只是垂直通過在內(nèi)啟動少量線程擴展,而不是水平通過集群擴展。三特性常用的生產(chǎn)的特性如下響應(yīng)式編程模型適用性內(nèi)嵌容器組件還有對日志消息測試及擴展等支持。 摘要: 原創(chuàng)出處 https://www.bysocket.com 「公眾號:泥瓦匠BYSocket 」歡迎關(guān)注和轉(zhuǎn)載,保留摘要,謝謝! 02:WebFlux 快速入門實踐 文章工程: JDK...

    gaara 評論0 收藏0
  • 分布式 - Jetty架構(gòu)

    摘要:實現(xiàn)為,即模式上一個依賴于下一個的調(diào)用,比如常見的就是這種模式。啟動實例化和設(shè)置運行時處理流程通常會實例化一個,注意的構(gòu)造方法最終會調(diào)用依次將構(gòu)成責(zé)任鏈因為這個連同都是類型。內(nèi)部所有等執(zhí)行即。 showImg(https://segmentfault.com/img/bV3OIg?w=326&h=314); showImg(https://segmentfault.com/img/bV...

    libin19890520 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<