pbootcms网站模板|日韩1区2区|织梦模板||网站源码|日韩1区2区|jquery建站特效-html5模板网

Spring多線程的使用以及問(wèn)題詳解

在我們開(kāi)發(fā)系統(tǒng)過(guò)程中,經(jīng)常會(huì)處理一些費(fèi)時(shí)間的任務(wù)(如:向數(shù)據(jù)庫(kù)中插入大量數(shù)據(jù)),這個(gè)時(shí)候就就需要使用多線程,下面這篇文章主要給大家介紹了關(guān)于Spring多線程的使用以及問(wèn)題的相關(guān)

前言

由于本周大部分時(shí)間都在寫(xiě)原型,主要遇到的問(wèn)題就是對(duì)實(shí)際功能理解不準(zhǔn)確導(dǎo)致多次修改原型浪費(fèi)了很多時(shí)間,這也就告訴我們一定要明確實(shí)際要求再去下手。

因?yàn)橹皶?huì)議中也多次提到了線程,而我本人對(duì)線程沒(méi)有什么理解于是便有了以下文章。

為什么使用多線程

在我們開(kāi)發(fā)系統(tǒng)過(guò)程中,經(jīng)常會(huì)處理一些費(fèi)時(shí)間的任務(wù)(如:向數(shù)據(jù)庫(kù)中插入大量數(shù)據(jù)),這個(gè)時(shí)候就就需要使用多線程。

Springboot中是否對(duì)多線程方法進(jìn)行了封裝

是,Spring中可直接由@Async實(shí)現(xiàn)多線程操作

如何控制線程運(yùn)行中的各項(xiàng)參數(shù)

通過(guò)配置線程池。

線程池ThreadPoolExecutor執(zhí)行規(guī)則如下

然后我們來(lái)認(rèn)為構(gòu)造一個(gè)線程池來(lái)試一下:

@Configuration
@EnableAsync
public class ThreadPoolConfig implements AsyncConfigurer {
  /**
   * 核心線程池大小
   */
  private static final int CORE_POOL_SIZE = 3;

  /**
   * 最大可創(chuàng)建的線程數(shù)
   */
  private static final int MAX_POOL_SIZE = 10;

  /**
   * 隊(duì)列最大長(zhǎng)度
   */
  private static final int QUEUE_CAPACITY = 10;

  /**
   * 線程池維護(hù)線程所允許的空閑時(shí)間
   */
  private static final int KEEP_ALIVE_SECONDS = 300;

  /**
   * 異步執(zhí)行方法線程池
   *
   * @return
   */
  @Override
  @Bean
  public Executor getAsyncExecutor() {
    ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
    executor.setMaxPoolSize(MAX_POOL_SIZE);
    executor.setCorePoolSize(CORE_POOL_SIZE);
    executor.setQueueCapacity(QUEUE_CAPACITY);
    executor.setKeepAliveSeconds(KEEP_ALIVE_SECONDS);
    executor.setThreadNamePrefix("LiMingTest");
    // 線程池對(duì)拒絕任務(wù)(無(wú)線程可用)的處理策略
    executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
    executor.initialize();
    return executor;
  }
}

ThreadPoolExecutor是JDK中的線程池實(shí)現(xiàn),這個(gè)類實(shí)現(xiàn)了一個(gè)線程池需要的各個(gè)方法,它提供了任務(wù)提交、線程管理、監(jiān)控等方法。

corePoolSize:核心線程數(shù)

線程池維護(hù)的最小線程數(shù)量,默認(rèn)情況下核心線程創(chuàng)建后不會(huì)被回收(注意:設(shè)置allowCoreThreadTimeout=true后,空閑的核心線程超過(guò)存活時(shí)間也會(huì)被回收)。

大于核心線程數(shù)的線程,在空閑時(shí)間超過(guò)keepAliveTime后會(huì)被回收。

maximumPoolSize:最大線程數(shù)

線程池允許創(chuàng)建的最大線程數(shù)量。

當(dāng)添加一個(gè)任務(wù)時(shí),核心線程數(shù)已滿,線程池還沒(méi)達(dá)到最大線程數(shù),并且沒(méi)有空閑線程,工作隊(duì)列已滿的情況下,創(chuàng)建一個(gè)新線程,然后從工作隊(duì)列的頭部取出一個(gè)任務(wù)交由新線程來(lái)處理,而將剛提交的任務(wù)放入工作隊(duì)列尾部。

keepAliveTime:空閑線程存活時(shí)間

當(dāng)一個(gè)可被回收的線程的空閑時(shí)間大于keepAliveTime,就會(huì)被回收。

被回收的線程:

設(shè)置allowCoreThreadTimeout=true的核心線程。
大于核心線程數(shù)的線程(非核心線程)。

workQueue:工作隊(duì)列

新任務(wù)被提交后,如果核心線程數(shù)已滿則會(huì)先添加到工作隊(duì)列,任務(wù)調(diào)度時(shí)再?gòu)年?duì)列中取出任務(wù)。工作隊(duì)列實(shí)現(xiàn)了BlockingQueue接口。

handler:拒絕策略

當(dāng)線程池線程數(shù)已滿,并且工作隊(duì)列達(dá)到限制,新提交的任務(wù)使用拒絕策略處理。可以自定義拒絕策略,拒絕策略需要實(shí)現(xiàn)RejectedExecutionHandler接口。

JDK默認(rèn)的拒絕策略有四種:

AbortPolicy:丟棄任務(wù)并拋出RejectedExecutionException異常。
DiscardPolicy:丟棄任務(wù),但是不拋出異常。可能導(dǎo)致無(wú)法發(fā)現(xiàn)系統(tǒng)的異常狀態(tài)。
DiscardOldestPolicy:丟棄隊(duì)列最前面的任務(wù),然后重新提交被拒絕的任務(wù)。
CallerRunsPolicy:由調(diào)用線程處理該任務(wù)。

我們?cè)诜菧y(cè)試文件中直接使用new Thread創(chuàng)建新線程時(shí)編譯器會(huì)發(fā)出警告:

不要顯式創(chuàng)建線程,請(qǐng)使用線程池。
說(shuō)明:使用線程池的好處是減少在創(chuàng)建和銷(xiāo)毀線程上所花的時(shí)間以及系統(tǒng)資源的開(kāi)銷(xiāo),解決資源不足的問(wèn)題。如果不使用線程池,有可能造成系統(tǒng)創(chuàng)建大量同類線程而導(dǎo)致消耗完內(nèi)存或者“過(guò)度切換”的問(wèn)題

public class TestServiceImpl implements TestService {
  private final static Logger logger = LoggerFactory.getLogger(TestServiceImpl.class);
  @Override
  public void task(int i) {
      logger.info("任務(wù): "+i);
  }
}
@Autowired
  TestService testService;
  @Test
  public void test() {
    for (int i = 0; i < 50; i++) {
      testService.task(i);
    }

我們可以看到一切執(zhí)行正常;

之后我有對(duì)線程進(jìn)行了一些測(cè)試:

class TestServiceImplTest {
  @Test
  public void test() {
    Thread add = new AddThread();
    Thread dec = new DecThread();
    add.start();
    dec.start();
    add.join();
    dec.join();
    System.out.println(Counter.count);
  }

  static class Counter {
    public static int count = 0;
  }

  class AddThread extends Thread {
    public void run() {
      for (int i=0; i<10000; i++) { Counter.count += 1; }
    }
  }

  class DecThread extends Thread {
    public void run() {
      for (int i=0; i<10000; i++) { Counter.count -= 1; }
    }
  }

一個(gè)自增線程,一個(gè)自減線程,對(duì)0進(jìn)行同樣次數(shù)的操作,理應(yīng)結(jié)果仍然為零,但是執(zhí)行結(jié)果卻每次都不同。

經(jīng)過(guò)搜索之后發(fā)現(xiàn)對(duì)變量進(jìn)行讀取和寫(xiě)入時(shí),結(jié)果要正確,必須保證是原子操作。原子操作是指不能被中斷的一個(gè)或一系列操作。

例如,對(duì)于語(yǔ)句: n +=1; 看似只有一行語(yǔ)句卻包括了3條指令:

讀取n, n+1, 存儲(chǔ)n;

比如有以下兩個(gè)進(jìn)程同時(shí)對(duì)10進(jìn)行加1操作

這說(shuō)明多線程模型下,要保證邏輯正確,對(duì)共享變量進(jìn)行讀寫(xiě)時(shí),必須保證一組指令以原子方式執(zhí)行:即某一個(gè)線程執(zhí)行時(shí),其他線程必須等待。

static class Counter {
    public static final Object lock = new Object();//每個(gè)線程都需獲得鎖才能執(zhí)行
    public static int count = 0;
  }

  class AddThread extends Thread {
    public void run() {
      for (int i=0; i<10000; i++) {
        synchronized(Counter.lock) { static class Counter {
    public static final Object lock = new Object();
    public static int count = 0;
  }

  class DecThread extends Thread {
    public void run() {
      for (int i=0; i<10000; i++) {
        synchronized(Counter.lock) {
          Counter.count -= 1;
        }
      }
    }
  }

值得注意的是每個(gè)類可以設(shè)置多個(gè)鎖,如果線程獲取的不是同一個(gè)鎖則無(wú)法起到上述功能;

springBoot中也定義了很多類型的鎖,在此就不一一說(shuō)明了,我們目前能做到的就是注意項(xiàng)目中的異步操作,觀察操作所使用的線程,做到在以后項(xiàng)目中遇到此類問(wèn)題時(shí)能及時(shí)發(fā)現(xiàn)問(wèn)題,解決問(wèn)題。

總結(jié)

到此這篇關(guān)于Spring多線程的使用及問(wèn)題的文章就介紹到這了,更多相關(guān)Spring多線程使用內(nèi)容請(qǐng)搜索html5模板網(wǎng)以前的文章希望大家以后多多支持html5模板網(wǎng)!

【網(wǎng)站聲明】本站部分內(nèi)容來(lái)源于互聯(lián)網(wǎng),旨在幫助大家更快的解決問(wèn)題,如果有圖片或者內(nèi)容侵犯了您的權(quán)益,請(qǐng)聯(lián)系我們刪除處理,感謝您的支持!

相關(guān)文檔推薦

主站蜘蛛池模板: 火锅底料批发-串串香技术培训[川禾川调官网] | 金属回收_废铜废铁回收_边角料回收_废不锈钢回收_废旧电缆线回收-广东益夫金属回收公司 | 金属回收_废铜废铁回收_边角料回收_废不锈钢回收_废旧电缆线回收-广东益夫金属回收公司 | 铝箔袋,铝箔袋厂家,东莞铝箔袋,防静电铝箔袋,防静电屏蔽袋,防静电真空袋,真空袋-东莞铭晋让您的产品与众不同 | 皮带机_移动皮带机_大倾角皮带机_皮带机厂家 - 新乡市国盛机械设备有限公司 | 企业微信营销_企业微信服务商_私域流量运营_艾客SCRM官网 | 湖南自考_湖南自学考试网| 防水套管|柔性防水套管|伸缩器|伸缩接头|传力接头-河南伟创管道 防水套管_柔性防水套管_刚性防水套管-巩义市润达管道设备制造有限公司 | pos机办理,智能/扫码/二维码/微信支付宝pos机-北京万汇通宝商贸有限公司 | 山东PE给水管厂家,山东双壁波纹管,山东钢带增强波纹管,山东PE穿线管,山东PE农田灌溉管,山东MPP电力保护套管-山东德诺塑业有限公司 | 代写标书-专业代做标书-商业计划书代写「深圳卓越创兴公司」 | 滑石粉,滑石粉厂家,超细滑石粉-莱州圣凯滑石有限公司 | 户外环保不锈钢垃圾桶_标识标牌制作_园林公园椅厂家_花箱定制-北京汇众环艺 | 热熔胶网膜|pes热熔网膜价格|eva热熔胶膜|热熔胶膜|tpu热熔胶膜厂家-苏州惠洋胶粘制品有限公司 | 防火窗_耐火窗_防火门厂家_防火卷帘门-重庆三乐门业有限公司 | 珠海白蚁防治_珠海灭鼠_珠海杀虫灭鼠_珠海灭蟑螂_珠海酒店消杀_珠海工厂杀虫灭鼠_立净虫控防治服务有限公司 | 滤芯,过滤器,滤油机,贺德克滤芯,精密滤芯_新乡市宇清流体净化技术有限公司 | 杭州中策电线|中策电缆|中策电线|杭州中策电缆|杭州中策电缆永通集团有限公司 | 捷码低代码平台 - 3D数字孪生_大数据可视化开发平台「免费体验」 | 四川成都干燥设备_回转筒干燥机_脉冲除尘器_输送设备_热风炉_成都川工星科机电设备有限公司 | 硬齿面减速机[型号全],ZQ减速机-淄博久增机械 | 澳门精准正版免费大全,2025新澳门全年免费,新澳天天开奖免费资料大全最新,新澳2025今晚开奖资料,新澳马今天最快最新图库-首页-东莞市傲马网络科技有限公司 | 铝箔袋,铝箔袋厂家,东莞铝箔袋,防静电铝箔袋,防静电屏蔽袋,防静电真空袋,真空袋-东莞铭晋让您的产品与众不同 | 压接机|高精度压接机|手动压接机|昆明可耐特科技有限公司[官网] 胶泥瓷砖胶,轻质粉刷石膏,嵌缝石膏厂家,腻子粉批发,永康家德兴,永康市家德兴建材厂 | 深圳APP开发公司_软件APP定制开发/外包制作-红匣子科技 | 合肥卓创建筑装饰,专业办公室装饰、商业空间装修与设计。 | 回转窑-水泥|石灰|冶金-巩义市瑞光金属制品有限责任公司 | 除湿机|工业除湿机|抽湿器|大型地下室车间仓库吊顶防爆除湿机|抽湿烘干房|新风除湿机|调温/降温除湿机|恒温恒湿机|加湿机-杭州川田电器有限公司 | 识禅_对禅的了解,从这里开始| 书法培训-高考书法艺考培训班-山东艺霖书法培训凭实力挺进央美 | 特种电缆厂家-硅橡胶耐高温电缆-耐低温补偿导线-安徽万邦特种电缆有限公司 | 工业制氮机_psa制氮机厂家-宏骁智能装备科技江苏有限公司 | 德国EA可编程直流电源_电子负载,中国台湾固纬直流电源_交流电源-苏州展文电子科技有限公司 | 合肥花魁情感婚姻咨询中心_挽回爱情_修复婚姻_恋爱指南 | 干粉砂浆设备_干混砂浆生产线_腻子粉加工设备_石膏抹灰砂浆生产成套设备厂家_干粉混合设备_砂子烘干机--郑州铭将机械设备有限公司 | 尼龙PA610树脂,尼龙PA612树脂,尼龙PA1010树脂,透明尼龙-谷骐科技【官网】 | 影像测量仪_三坐标测量机_一键式二次元_全自动影像测量仪-广东妙机精密科技股份有限公司 | 深圳侦探联系方式_深圳小三调查取证公司_深圳小三分离机构 | 新型锤式破碎机_新型圆锥式_新型颚式破碎机_反击式打沙机_锤式制砂机_青州建源机械 | 渣土车电机,太阳能跟踪器电机,蜗轮蜗杆减速电机厂家-淄博传强电机 | 塑料托盘厂家直销-吹塑托盘生产厂家-力库塑业【官网】 |