一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Java實現終止線程池中正在運行的定時任務

Java實現終止線程池中正在運行的定時任務

2021-04-25 12:54hchstudio Java教程

本篇文章給大家分享了JAVA中實現終止線程池中正在運行的定時任務的具體步驟和方法,有需要的朋友跟著學習下。

最近項目中遇到了一個新的需求,就是實現一個可以動態添加定時任務的功能。說到這里,有人可能會說簡單啊,使用quartz就好了,簡單粗暴。然而quartz框架太重了,小項目根本不好操作啊。當然,也有人會說,jdk提供了timer的接口啊,完全夠用啊。但是我們項目的需求完全是多線程的模型啊,而timer是單線程的,so,樓主最后還是選擇了jdk的線程池

線程池是什么

Java通過Executors提供四種線程池,分別為:
newCachedThreadPool :創建一個可緩存線程池,如果線程池長度超過處理需要,可靈活回收空閑線程,若無可回收,則新建線程。
newFixedThreadPool : 創建一個定長線程池,可控制線程最大并發數,超出的線程會在隊列中等待。
newScheduledThreadPool : 創建一個定長線程池,支持定時及周期性任務執行。
newSingleThreadExecutor : 創建一個單線程化的線程池,它只會用唯一的工作線程來執行任務,保證所有任務按照指定順序(FIFO, LIFO, 優先級)執行。

樓主項目中用到的是newScheduledThreadPool, 就這些吧,再多的樓主就班門弄斧了,Google一下,一大堆。

線程池service的獲取

樓主通過單例模式來獲取線程池的service,代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
 * 線程池創建.
 * @author wuhf
 * @date 2018/01/16
 */
public class ThreadPoolUtils {
 
  private static ScheduledExecutorService executorService;
 
  private ThreadPoolUtils() {
    //手動創建線程池.
    executorService = new ScheduledThreadPoolExecutor(10,
        new BasicThreadFactory.Builder().namingPattern("syncdata-schedule-pool-%d").daemon(true).build());
  }
 
  private static class PluginConfigHolder {
    private final static ThreadPoolUtils INSTANCE = new ThreadPoolUtils();
  }
 
  public static ThreadPoolUtils getInstance() {
    return PluginConfigHolder.INSTANCE;
  }
 
  public ScheduledExecutorService getThreadPool(){
    return executorService;
  }
 
}

中斷某一個正在運行的線程代碼實現

廢話就不多說了,代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
/**
 * 中斷線程池的某個任務.
 */
public class InterruptThread implements Runnable {
 
  private int num;
 
  public InterruptThread (int num){
    this.num = num;
  }
 
  public static void main(String[] args) throws InterruptedException {
 
    Thread interruptThread = new Thread(new InterruptThread(1));
    ScheduledFuture<?> t = ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread,0,2,
        TimeUnit.SECONDS);
 
    InterruptThread interruptThread1 = new InterruptThread(2);
    ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread1,0,2,
        TimeUnit.SECONDS);
 
    InterruptThread interruptThread2 = new InterruptThread(3);
    ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread2,0,2,
        TimeUnit.SECONDS);
    Thread.sleep(5000);
 
        //終止正在運行的線程interruptThread
    t.cancel(true);
    while (true){
 
    }
  }
 
  @Override
  public void run() {
    System.out.println("this is a thread" + num);
  }
}

踩坑記錄

樓主在使用如下代碼時,突然想到當這個定時任務需要被停止時該如何停止線程運行

?
1
2
ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread,0,2,
        TimeUnit.SECONDS);

既然我有這樣的需求,那就Google一下吧,找了大半圈,愣是沒找到相關資料,都是一些關于Java線程池的深入分析。或者是全局變量啥的,并沒有找到令樓主滿意的解決方案。

既然沒有線程的那就扒一下scheduleAtFixedRate的底層源碼看看是什么東西吧,果不其然我在源碼中看到了scheduleAtFixedRate方法的具體實現,發現他的返回值是ScheduledFuture。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public ScheduledFuture<?> scheduleAtFixedRate(Runnable command,
                         long initialDelay,
                         long period,
                         TimeUnit unit) {
    if (command == null || unit == null)
      throw new NullPointerException();
    if (period <= 0)
      throw new IllegalArgumentException();
    ScheduledFutureTask<Void> sft =
      new ScheduledFutureTask<Void>(command,
                     null,
                     triggerTime(initialDelay, unit),
                     unit.toNanos(period));
    RunnableScheduledFuture<Void> t = decorateTask(command, sft);
    sft.outerTask = t;
    delayedExecute(t);
    return t;
  }

接著往下我們再看看ScheduledFuture里面有什么東西吧,沒有讓樓主失望,看到了這個

?
1
2
3
4
5
6
7
8
9
10
11
12
13
public boolean cancel(boolean mayInterruptIfRunning) {
      boolean cancelled = super.cancel(mayInterruptIfRunning);
      if (cancelled && removeOnCancel && heapIndex >= 0)
        remove(this);
      return cancelled;
}
      
//從線程的運行隊列中移除當前線程
public boolean remove(Runnable task) {
    boolean removed = workQueue.remove(task);
    tryTerminate(); // In case SHUTDOWN and now empty
    return removed;
}

再往上查super.cancel(mayInterruptIfRunning)是什么東西,我們看到了這個,

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//通過調用線程的interrupt方法終止線程運行
public boolean cancel(boolean mayInterruptIfRunning) {
    if (!(state == NEW &&
       UNSAFE.compareAndSwapInt(this, stateOffset, NEW,
         mayInterruptIfRunning ? INTERRUPTING : CANCELLED)))
      return false;
    try // in case call to interrupt throws exception
      if (mayInterruptIfRunning) {
        try {
          Thread t = runner;
          if (t != null)
            t.interrupt();
        } finally { // final state
          UNSAFE.putOrderedInt(this, stateOffset, INTERRUPTED);
        }
      }
    } finally {
      finishCompletion();
    }
    return true;
  }

到這里所有的問題都迎刃而解。

總結一下吧

項目中總是會遇到比較難搞的解決方案,當Google不太好找時,翻一下jdk的源碼或許也是一個不錯的方法。

原文鏈接:http://www.hchstudio.cn/article/2018/cc2a/

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 好骚好紧 | 男人和女人上床 | caoporen在线视频入口 | 精品视频免费在线观看 | 亚洲欧美专区 | 日本连裤袜xxxxx在线视频 | 短篇同学新婚h系列小说 | 操b图片| 香蕉国产成版人视频在线观看 | 强行扒开美女大腿挺进 | 99精品国产综合久久久久 | 国产欧美亚洲精品第一页青草 | 欧美日韩精品免费一区二区三区 | www.四虎网站 | 饭冈加奈子黑人解禁在线播放 | 四虎精品永久在线网址 | 男人晚上适合偷偷看的污污 | 日韩精品一区二区三区中文在线 | 丰满的闺蜜2中文字幕 | 日本妇人成熟免费观看18 | 色久久一个亚洲综合网 | 亚洲狠狠婷婷综合久久久久网站 | 国产视频分类 | 紧身牛仔裤美女被啪啪久久网 | 国产精品福利在线观看入口 | 国产精品成人一区二区1 | 日韩欧美亚洲一区精选 | 国产午夜免费视频 | 欧美草逼网站 | 色先锋 影音先锋a 资源站 | 亚洲欧美日韩天堂 | 91中文在线 | 日本在线色 | 91啪在线观看国产在线 | 国产色视频一区二区三区 | 国产亚洲高清国产拍精品 | 久久免费看少妇高潮A片特爽 | 美国video | 扒开女人下面使劲桶屁股动漫 | 国产精品suv| 欧美人与禽杂交大片 |