成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

Tomcat使用線程池處理遠程并發請求的方法

瀏覽:259日期:2023-03-19 16:51:28

通過了解學習tomcat如何處理并發請求,了解到線程池,鎖,隊列,unsafe類,下面的主要代碼來自

java-jre:

sun.misc.Unsafe
java.util.concurrent.ThreadPoolExecutor
java.util.concurrent.ThreadPoolExecutor.Worker
java.util.concurrent.locks.AbstractQueuedSynchronizer
java.util.concurrent.locks.AbstractQueuedLongSynchronizer
java.util.concurrent.LinkedBlockingQueue

tomcat:

org.apache.tomcat.util.net.NioEndpoint
org.apache.tomcat.util.threads.ThreadPoolExecutor
org.apache.tomcat.util.threads.TaskThreadFactory
org.apache.tomcat.util.threads.TaskQueue

ThreadPoolExecutor

是一個線程池實現類,管理線程,減少線程開銷,可以用來提高任務執行效率,

構造方法中的參數有

public ThreadPoolExecutor(
 int corePoolSize,
 int maximumPoolSize,
 long keepAliveTime,
 TimeUnit unit,
 BlockingQueue<Runnable> workQueue,
 ThreadFactory threadFactory,
 RejectedExecutionHandler handler) {
 
}

corePoolSize 是核心線程數
maximumPoolSize 是最大線程數
keepAliveTime 非核心線程最大空閑時間(超過時間終止)
unit 時間單位
workQueue 隊列,當任務過多時,先存放在隊列
threadFactory 線程工廠,創建線程的工廠
handler 決絕策略,當任務數過多,隊列不能再存放任務時,該如何處理,由此對象去處理。這是個接口,你可以自定義處理方式

ThreadPoolExecutor在Tomcat中http請求的應用

此線程池是tomcat用來在接收到遠程請求后,將每次請求單獨作為一個任務去處理,每次調用execute(Runnable)

初始化

org.apache.tomcat.util.net.NioEndpoint

NioEndpoint初始化的時候,創建了線程池

public void createExecutor() {
 internalExecutor = true;
 TaskQueue taskqueue = new TaskQueue();
 //TaskQueue無界隊列,可以一直添加,因此handler 等同于無效
 TaskThreadFactory tf = new TaskThreadFactory(getName() + "-exec-", daemon, getThreadPriority());
 executor = new ThreadPoolExecutor(getMinSpareThreads(), getMaxThreads(), 60, TimeUnit.SECONDS,taskqueue, tf);
 taskqueue.setParent( (ThreadPoolExecutor) executor);
 }

在線程池創建時,調用prestartAllCoreThreads(), 初始化核心工作線程worker,并啟動

public int prestartAllCoreThreads() {
 int n = 0;
 while (addWorker(null, true))
  ++n;
 return n;
 }

當addWorker 數量等于corePoolSize時,addWorker(null,ture)會返回false,停止worker工作線程的創建

提交任務到隊列

每次客戶端過來請求(http),就會提交一次處理任務,

worker 從隊列中獲取任務運行,下面是任務放入隊列的邏輯代碼

ThreadPoolExecutor.execute(Runnable) 提交任務:

public void execute(Runnable command) {
 if (command == null)
  throw new NullPointerException();
 
 int c = ctl.get();
 	// worker數 是否小于 核心線程數 tomcat中初始化后,一般不滿足第一個條件,不會addWorker
 if (workerCountOf(c) < corePoolSize) {
  if (addWorker(command, true))
  return;
  c = ctl.get();
 }
 	// workQueue.offer(command),將任務添加到隊列,
 if (isRunning(c) && workQueue.offer(command)) {
  int recheck = ctl.get();
  if (! isRunning(recheck) && remove(command))
  reject(command);
  else if (workerCountOf(recheck) == 0)
  addWorker(null, false);
 }
 else if (!addWorker(command, false))
  reject(command);
 }

workQueue.offer(command) 完成了任務的提交(在tomcat處理遠程http請求時)。

workQueue.offer

TaskQueue 是 BlockingQueue 具體實現類,workQueue.offer(command)實際代碼:

public boolean offer(E e) {
 if (e == null) throw new NullPointerException();
 final AtomicInteger count = this.count;
 if (count.get() == capacity)
 return false;
 int c = -1;
 Node<E> node = new Node<E>(e);
 final ReentrantLock putLock = this.putLock;
 putLock.lock();
 try {
 if (count.get() < capacity) {
  enqueue(node); //此處將任務添加到隊列
  c = count.getAndIncrement();
  if (c + 1 < capacity)
  notFull.signal();
 }
 } finally {
 putLock.unlock();
 }
 if (c == 0)
 signalNotEmpty();
 return c >= 0;
}

// 添加任務到隊列
/**
 * Links node at end of queue.
 *
 * @param node the node
 */
private void enqueue(Node<E> node) {
 // assert putLock.isHeldByCurrentThread();
 // assert last.next == null;
 last = last.next = node; //鏈表結構 last.next = node; last = node
}

之后是worker的工作,worker在run方法中通過去getTask()獲取此處提交的任務,并執行完成任務。

線程池如何處理新提交的任務

添加worker之后,提交任務,因為worker數量達到corePoolSize,任務都會將放入隊列,而worker的run方法則是循環獲取隊列中的任務(不為空時),

worker run方法:

/** Delegates main run loop to outer runWorker */
 public void run() {
  runWorker(this);
 }

循環獲取隊列中的任務

runWorker(worker)方法 循環部分代碼:

final void runWorker(Worker w) {
 Thread wt = Thread.currentThread();
 Runnable task = w.firstTask;
 w.firstTask = null;
 w.unlock(); // allow interrupts
 boolean completedAbruptly = true;
 try {
  while (task != null || (task = getTask()) != null) { //循環獲取隊列中的任務
  w.lock(); // 上鎖
  try {
   // 運行前處理
   beforeExecute(wt, task);
   // 隊列中的任務開始執行
   task.run();
   // 運行后處理
   afterExecute(task, thrown);
  } finally {
   task = null;
   w.completedTasks++;
   w.unlock(); // 釋放鎖
  }
  }
  completedAbruptly = false;
 } finally {
  processWorkerExit(w, completedAbruptly);
 }
 }

task.run()執行任務

鎖運用

ThreadPoolExecutor 使用鎖主要保證兩件事情,
1.給隊列添加任務,保證其他線程不能操作隊列
2.獲取隊列的任務,保證其他線程不能同時操作隊列

給隊列添加任務上鎖

public boolean offer(E e) {
 if (e == null) throw new NullPointerException();
 final AtomicInteger count = this.count;
 if (count.get() == capacity)
  return false;
 int c = -1;
 Node<E> node = new Node<E>(e);
 final ReentrantLock putLock = this.putLock;
 putLock.lock(); //上鎖
 try {
  if (count.get() < capacity) {
  enqueue(node);
  c = count.getAndIncrement();
  if (c + 1 < capacity)
   notFull.signal();
  }
 } finally {
  putLock.unlock(); //釋放鎖
 }
 if (c == 0)
  signalNotEmpty();
 return c >= 0;
 }

 

獲取隊列任務上鎖

private Runnable getTask() {
 boolean timedOut = false; // Did the last poll() time out?
		// ...省略
 for (;;) {
  try {
  Runnable r = timed ?
   workQueue.poll(keepAliveTime, TimeUnit.NANOSECONDS) :
   workQueue.take(); //獲取隊列中一個任務
  if (r != null)
   return r;
  timedOut = true;
  } catch (InterruptedException retry) {
  timedOut = false;
  }
 }
 }
public E take() throws InterruptedException {
 E x;
 int c = -1;
 final AtomicInteger count = this.count;
 final ReentrantLock takeLock = this.takeLock;
 takeLock.lockInterruptibly(); // 上鎖
 try {
  while (count.get() == 0) {
  notEmpty.await(); //如果隊列中沒有任務,等待
  }
  x = dequeue();
  c = count.getAndDecrement();
  if (c > 1)
  notEmpty.signal();
 } finally {
  takeLock.unlock(); // 釋放鎖
 }
 if (c == capacity)
  signalNotFull();
 return x;
 }

volatile

在并發場景這個關鍵字修飾成員變量很常見,

主要目的公共變量在被某一個線程修改時,對其他線程可見(實時)

sun.misc.Unsafe 高并發相關類

線程池使用中,有平凡用到Unsafe類,這個類在高并發中,能做一些原子CAS操作,鎖線程,釋放線程等。

sun.misc.Unsafe 類是底層類,openjdk源碼中有

原子操作數據

java.util.concurrent.locks.AbstractQueuedSynchronizer 類中就有保證原子操作的代碼

protected final boolean compareAndSetState(int expect, int update) {
 // See below for intrinsics setup to support this
 return unsafe.compareAndSwapInt(this, stateOffset, expect, update);
 }

對應Unsafe類的代碼:

//對應的java底層,實際是native方法,對應C++代碼
/**
* Atomically update Java variable to <tt>x</tt> if it is currently
* holding <tt>expected</tt>.
* @return <tt>true</tt> if successful
*/
public final native boolean compareAndSwapInt(Object o, long offset,
      int expected,
      int x);

方法的作用簡單來說就是 更新一個值,保證原子性操作
當你要操作一個對象o的一個成員變量offset時,修改o.offset,
高并發下為保證準確性,你在操作o.offset的時候,讀應該是正確的值,并且中間不能被別的線程修改來保證高并發的環境數據操作有效。

即 expected 期望值與內存中的值比較是一樣的expected == 內存中的值 ,則更新值為 x,返回true代表修改成功

否則,期望值與內存值不同,說明值被其他線程修改過,不能更新值為x,并返回false,告訴操作者此次原子性修改失敗。

阻塞和喚醒線程

public native void park(boolean isAbsolute, long time); //阻塞當前線程

線程池的worker角色循環獲取隊列任務,如果隊列中沒有任務,worker.run 還是在等待的,不會退出線程,代碼中用了notEmpty.await() 中斷此worker線程,放入一個等待線程隊列(區別去任務隊列);當有新任務需要時,再notEmpty.signal()喚醒此線程

底層分別是
unsafe.park() 阻塞當前線程
public native void park(boolean isAbsolute, long time);

unsafe.unpark() 喚醒線程
public native void unpark(Object thread);

這個操作是對應的,阻塞時,先將thread放入隊列,喚醒時,從隊列拿出被阻塞的線程,unsafe.unpark(thread)喚醒指定線程。

java.util.concurrent.locks.AbstractQueuedLongSynchronizer.ConditionObject 類中

通過鏈表存放線程信息

// 添加一個阻塞線程
private Node addConditionWaiter() {
  Node t = lastWaiter;
  // If lastWaiter is cancelled, clean out.
  if (t != null && t.waitStatus != Node.CONDITION) {
  unlinkCancelledWaiters();
  t = lastWaiter;
  }
  Node node = new Node(Thread.currentThread(), Node.CONDITION);
  if (t == null)
  firstWaiter = node;
  else
  t.nextWaiter = node;
  lastWaiter = node; //將新阻塞的線程放到鏈表尾部
  return node;
 }

// 拿出一個被阻塞的線程
 public final void signal() {
  if (!isHeldExclusively())
  throw new IllegalMonitorStateException();
  Node first = firstWaiter; //鏈表中第一個阻塞的線程
  if (first != null)
  doSignal(first);
 }

// 拿到后,喚醒此線程
final boolean transferForSignal(Node node) {
  LockSupport.unpark(node.thread);
 return true;
 }
public static void unpark(Thread thread) {
 if (thread != null)
  UNSAFE.unpark(thread);
 }

到此這篇關于Tomcat使用線程池處理遠程并發請求的方法的文章就介紹到這了,更多相關Tomcat線程池處理遠程并發請求內容請搜索以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持!

標簽: Tomcat
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
亚洲国产wwwccc36天堂| 欧美一区二区三区婷婷月色| 久久一区二区三区四区五区| 亚洲欧洲日韩av| 欧美精品18| 久久蜜桃av一区精品变态类天堂| 大尺度一区二区| 制服丝袜激情欧洲亚洲| 国产一区二区三区不卡在线观看| 欧美色图天堂网| 久久国产精品99久久人人澡| 欧美日韩国产另类不卡| 蜜桃91丨九色丨蝌蚪91桃色| 在线观看日韩毛片| 麻豆精品久久精品色综合| 色噜噜狠狠一区二区三区果冻| 亚洲国产精品一区二区久久恐怖片 | av亚洲精华国产精华精| 日韩精品在线看片z| 99久久综合色| 国产日产欧产精品推荐色 | 欧美日韩国产色站一区二区三区| 六月丁香婷婷色狠狠久久| 欧美日韩在线精品一区二区三区激情| 久久精品国产在热久久| 欧美色综合影院| 国产精品一区二区在线观看网站| 91精品国产全国免费观看 | 亚洲成a人片在线不卡一二三区| 国产婷婷精品| 天天做天天摸天天爽国产一区| 久久一区二区三区av| 精品一区二区影视| 欧美一级黄色片| 99久久精品国产麻豆演员表| 国产欧美中文在线| 在线欧美视频| 亚洲韩国一区二区三区| 色老头久久综合| 激情小说亚洲一区| 精品裸体舞一区二区三区| 国产精品草草| 一区二区三区欧美日| 91国偷自产一区二区开放时间| 激情国产一区二区| 久久丝袜美腿综合| 黄色欧美成人| 青青草国产精品97视觉盛宴| 日韩一级免费一区| 国产字幕视频一区二区| 亚洲欧美日韩一区二区 | 青青草视频一区| 欧美一区二区黄色| 欧美福利一区二区三区| 亚洲欧美日韩国产综合在线| 久久婷婷激情| 国产福利一区二区三区视频在线 | 91高清视频免费看| 国产激情一区二区三区| 国产亚洲精品资源在线26u| 亚洲国产日韩欧美| 亚洲成人777| 制服丝袜亚洲网站| 欧美日韩日本网| 亚洲一区欧美一区| 69久久夜色精品国产69蝌蚪网| 成人性视频免费网站| 18涩涩午夜精品.www| 色欧美片视频在线观看在线视频| 国产精品一区在线| 国产精品久久久久久久久快鸭 | 国产精品入口麻豆原神| 另类av一区二区| 成人av午夜电影| 亚洲精品国产高清久久伦理二区| 欧美日韩aaaaa| 国产精品99一区二区| 日韩电影免费在线| xnxx国产精品| 性色一区二区三区| 成人av手机在线观看| 亚洲精品videosex极品| 欧美精品在线一区二区三区| 伊人成人在线视频| 国产美女精品人人做人人爽| 中文字幕在线观看不卡| 在线观看欧美精品| 欧美 日韩 国产一区二区在线视频| 亚洲444eee在线观看| 日韩欧美国产系列| 中文有码久久| 顶级嫩模精品视频在线看| 一区二区三区四区在线| 日韩欧美一区二区三区在线| 国产精品久久亚洲7777| 成人一道本在线| 午夜精品福利在线| 久久久久国产精品人| 在线一区二区三区做爰视频网站| 欧美黄色大片网站| 精品一区二区精品| 亚洲精品成人精品456| 日韩欧美国产电影| 久久综合久久久| 午夜久久99| 国产乱人伦偷精品视频免下载| 亚洲人成网站精品片在线观看| 欧美精品日日鲁夜夜添| 一本色道久久综合亚洲精品高清| heyzo一本久久综合| 免费人成黄页网站在线一区二区| 国产精品热久久久久夜色精品三区| 欧美挠脚心视频网站| 夜夜爽www精品| 99国产精品久久久| 韩国精品主播一区二区在线观看 | 亚洲一区日韩在线| 99久久婷婷国产综合精品电影| 日韩av电影免费观看高清完整版 | 欧美日韩国产小视频| 亚洲精品在线二区| 国产成a人亚洲| 日韩精品一级二级| 中文字幕中文在线不卡住| 日韩一区二区免费在线电影| 久久精品免费| 亚洲激情网站| 99久久精品费精品国产一区二区| 久久99九九99精品| 亚洲高清不卡在线观看| 国产精品另类一区| 欧美一区二区三区在线视频| 久久久久久九九九九| 亚洲性视频h| 99久久夜色精品国产网站| 精品在线观看免费| 亚洲一区二区在线免费观看视频 | 欧美高清性hdvideosex| 久久久久久久久久久久久9999| 国自产拍偷拍福利精品免费一| 福利电影一区二区| 久久福利视频一区二区| 午夜电影网一区| 亚洲一区欧美一区| 一区二区在线观看av| 中文字幕亚洲在| 国产视频一区在线观看| 精品区一区二区| 欧美日本国产一区| 日本高清免费不卡视频| 国产欧美日韩| 亚洲激情网址| 99精品99| 一区二区福利| 一区二区三区四区国产| 亚洲精品久久| 亚洲国产欧美日韩| 亚洲激情精品| 黄色国产精品一区二区三区| 欧美性事免费在线观看| 欧美日韩第一区| 欧美一区三区二区在线观看| 成人免费视频视频在线观看免费| 国产一区二区三区美女| 国产一区二区在线影院| 久久91精品国产91久久小草| 久久99这里只有精品| 蜜臀av一区二区三区| 日本不卡视频在线观看| 日韩在线观看一区二区| 日韩一区欧美二区| 蜜臀va亚洲va欧美va天堂| 美女高潮久久久| 久久不见久久见免费视频7| 蜜臀精品久久久久久蜜臀| 日韩av一区二区在线影视| 日韩电影一区二区三区| 免费欧美日韩国产三级电影| 日韩激情一区二区| 青青草国产成人99久久| 日本不卡一区二区三区| 男男gaygay亚洲| 激情六月婷婷综合| 国产成人精品免费网站| 成人免费va视频| 91日韩一区二区三区| 欧美视频在线观看| 亚洲第一区色| 国产视频欧美| 91国偷自产一区二区开放时间 | 欧美精品在线观看一区二区| 欧美精品在线视频| 欧美成人a∨高清免费观看| 26uuu国产电影一区二区| 国产日产亚洲精品系列| 中文字幕一区二区三中文字幕| 亚洲综合色丁香婷婷六月图片| 日韩精品亚洲专区| 国产在线一区二区综合免费视频| 国产大陆精品国产|