<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          SpringBoot + 虛擬線程,性能炸裂!

          共 4189字,需瀏覽 9分鐘

           ·

          2024-06-03 22:49

          什么是虛擬線程

          虛擬線程是Java19開(kāi)始增加的一個(gè)特性,和Golang的攜程類似,一個(gè)其它語(yǔ)言早就提供的、且如此實(shí)用且好用的功能,作為一個(gè)Java開(kāi)發(fā)者,早就已經(jīng)望眼欲穿了。

          虛擬線程和普通線程的區(qū)別

          “虛擬”線程,望文生義,它是“假”的,它不直接調(diào)度操作系統(tǒng)的線程,而是由JVM再提供一層線程的接口抽象,由普通線程調(diào)度,即一個(gè)普通的操作系統(tǒng)線程可以調(diào)度成千上萬(wàn)個(gè)虛擬線程。

          虛擬線程比普通線程的消耗要小得多得多,在內(nèi)存足夠的情況下,我們甚至可以創(chuàng)建上百萬(wàn)的虛擬線程,這在之前(Java19以前)是不可能的。

          ?

          其實(shí)如果有用過(guò)akka的朋友們會(huì)發(fā)現(xiàn),其實(shí)兩者很相似,只不過(guò)使用akka是應(yīng)用程序來(lái)處理,而虛擬線程是JVM來(lái)處理,使用上更簡(jiǎn)潔且方便。

          ?

          SpringBoot使用虛擬線程

          下面我們會(huì)在SpringBoot中使用虛擬線程,將默認(rèn)的異步線程池和http處理線程池替換為虛擬線程,然后對(duì)比虛擬線程和普通線程的性能差異,你會(huì)發(fā)現(xiàn)差別就像馬車換高鐵,不是一個(gè)時(shí)代的東西。

          配置

          首先我們使用的Java版本是java-20.0.2-oracle,SpringBoot版本是3.1.2。

          要在SpringBoot中使用虛擬線程很簡(jiǎn)單,增加如下配置即可:

          /**
           * 配置是用于稍后測(cè)試,spring.virtual-thread=true是使用虛擬線程,false時(shí)還是使用默認(rèn)的普通線程
           */

          @Configuration
          @ConditionalOnProperty(prefix = "spring", name = "virtual-thread", havingValue = "true")
          public class ThreadConfig {

              @Bean
              public AsyncTaskExecutor applicationTaskExecutor() {
                  return new TaskExecutorAdapter(Executors.newVirtualThreadPerTaskExecutor());
              }

              @Bean
              public TomcatProtocolHandlerCustomizer<?> protocolHandlerCustomizer() {
                  return protocolHandler -> {
                      protocolHandler.setExecutor(Executors.newVirtualThreadPerTaskExecutor());
                  };
              }
          }

          @Async性能對(duì)比

          我們寫(xiě)一個(gè)異步service,里面睡眠50ms,模擬MySQL或Redis等IO操作:

          @Service
          public class AsyncService {

              /**
               * 
               * @param countDownLatch 用于測(cè)試
               */

              @Async
              public void doSomething(CountDownLatch countDownLatch) throws InterruptedException {
                  Thread.sleep(50);
                  countDownLatch.countDown();
              }
          }

          最后測(cè)試類,很簡(jiǎn)單,就是循環(huán)調(diào)用這個(gè)方法10萬(wàn)次,計(jì)算所有方法執(zhí)行完成的消耗的時(shí)間:

          @Test
          public void testAsync() throws InterruptedException {
              long start = System.currentTimeMillis();
              int n = 100000;
              CountDownLatch countDownLatch = new CountDownLatch(n);
              for (int i = 0; i < n; i++) {
                  asyncService.doSomething(countDownLatch);
              }
              countDownLatch.await();
              long end = System.currentTimeMillis();
              System.out.println("耗時(shí):" + (end - start) + "ms");
          }

          普通線程耗時(shí):678秒左右,超過(guò)10分鐘了

          虛擬線程耗時(shí):3.9秒!!

          朋友們,接近200倍的性能差距!!

          HTTP請(qǐng)求性能對(duì)比

          讓我們?cè)倏纯磆ttp請(qǐng)求的對(duì)比,簡(jiǎn)單寫(xiě)個(gè)get請(qǐng)求,里面什么也不做,一樣睡50ms,模擬IO操作:

          @RequestMapping("/get")
          public Object get() throws Exception {
              Thread.sleep(50);
              return "ok";
          }

          然后我們使用jmeter請(qǐng)求接口,500個(gè)并發(fā)線程,運(yùn)行1萬(wàn)次,看看效果如何:

          「普通線程:」

          可以看到最小用時(shí)50ms,這個(gè)沒(méi)毛病,接口里面睡眠了50ms,但是不管是中位數(shù)還是90/95/99線都大于150ms了,這是因?yàn)橄到y(tǒng)線程是一個(gè)很昂貴的資源,SpringBoot中tomcat默認(rèn)的最大連接數(shù)應(yīng)該是200,在連接池的線程被耗盡后,這200個(gè)線程在那干等50ms結(jié)束,而剩下的請(qǐng)求也只能等待,無(wú)法進(jìn)行其它的操作。下面再看下虛擬線程的表現(xiàn):

          「虛擬線程耗時(shí):」

          可以看到即使是最大耗時(shí),也保持在100ms以下,即線程等待時(shí)間顯著的減少,虛擬線程更好的利用了系統(tǒng)資源。

          總結(jié)

          從上面的性能對(duì)比來(lái)看,虛擬線程在性能方面有明顯的優(yōu)勢(shì),但是要注意的是,我們上面的測(cè)試都是讓線程等待了50ms,這是模擬什么場(chǎng)景?

          沒(méi)錯(cuò),是IO密集型場(chǎng)景,即線程大部分時(shí)間是在等待IO,這樣虛擬線程才可以發(fā)揮出它的優(yōu)勢(shì),如果是CPU密集型場(chǎng)景,那么可能效果并不大。不過(guò)我們目前大部分的應(yīng)用都是IO密集型應(yīng)用較多,比如典型的WEB應(yīng)用,大量的時(shí)間在等待網(wǎng)絡(luò)IO(DB、緩存、HTTP等等),使用虛擬線程的效果還是非常明顯的。

          ?

          最后:大部分的公司可能還在用Java8,但是我想說(shuō)的是,是時(shí)候升級(jí)了,跟上時(shí)代的腳步吧,朋友們!

          ?
          來(lái)源:juejin.cn/post/7266745788536799247
              
          后端專屬技術(shù)群

          以構(gòu)建高質(zhì)量的技術(shù)交流社群為目的,歡迎從事編程開(kāi)發(fā)、技術(shù)招聘HR進(jìn)群,分享內(nèi)推信息技術(shù)交流,相互幫助,一起進(jìn)步!

          廣告人士勿入,切勿輕信私聊,防止被騙

          瀏覽 106
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  人人做天天爱夜夜 | 亚洲AV秘 无码一区二区三竹菊 | 一级a一级a爱片免费视频 | 亚洲秘 无码一区二区三区电影 | 夜夜看,夜夜爽 |