欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JVM系列之String.intern的性能解析

 更新時間:2022年05月18日 16:27:11   作者:flydean  
這篇文章主要介紹了JVM系列之String.intern的性能解析,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下

String對象有個特殊的StringTable字符串常量池,為了減少Heap中生成的字符串的數(shù)量,推薦盡量直接使用String Table中的字符串常量池中的元素。

那么String.intern的性能怎么樣呢?我們一起來看一下。

String.intern和G1字符串去重的區(qū)別

之前我們提到了,String.intern方法會返回字符串常量池中的字符串對象的引用。

而G1垃圾回收器的字符串去重的功能其實和String.intern有點不一樣,G1是讓兩個字符串的底層指向同一個byte[]數(shù)組。

有圖為證:

上圖中的String1和String2指向的是同一個byte[]數(shù)組。

String.intern的性能

我們看下intern方法的定義:

public native String intern();

大家可以看到這是一個native的方法。native底層肯定是C++實現(xiàn)的。

那么是不是native方法一定會比java方法快呢?

其實native方法有這樣幾個耗時點:

  1. native方法需要調(diào)用JDK-JVM接口,實際上是會浪費時間的。
  2. 性能會受到native方法中HashTable實現(xiàn)方法的制約,如果在高并發(fā)的情況下,native的HashTable的實現(xiàn)可能成為性能的制約因素。

舉個例子

還是用JMH工具來進行性能分析,我們使用String.intern,HashMap,和ConcurrentHashMap來對比分析,分別調(diào)用1次,100次,10000次和1000000。

代碼如下:

@State(Scope.Benchmark)
@BenchmarkMode(Mode.AverageTime)
@OutputTimeUnit(TimeUnit.NANOSECONDS)
@Fork(value = 1, jvmArgsPrepend = "-XX:+PrintStringTableStatistics")
@Warmup(iterations = 5)
@Measurement(iterations = 5)
public class StringInternBenchMark {
  @Param({"1", "100", "10000", "1000000"})
  private int size;
  private StringInterner str;
  private ConcurrentHashMapInterner chm;
  private HashMapInterner hm;
  @Setup
  public void setup() {
    str = new StringInterner();
    chm = new ConcurrentHashMapInterner();
    hm = new HashMapInterner();
  }
  public static class StringInterner {
    public String intern(String s) {
      return s.intern();
    }
  }
  @Benchmark
  public void useIntern(Blackhole bh) {
    for (int c = 0; c < size; c++) {
      bh.consume(str.intern("doit" + c));
    }
  }
  public static class ConcurrentHashMapInterner {
    private final Map<String, String> map;
    public ConcurrentHashMapInterner() {
      map = new ConcurrentHashMap<>();
    }
    public String intern(String s) {
      String exist = map.putIfAbsent(s, s);
      return (exist == null) ? s : exist;
    }
  }
  @Benchmark
  public void useCurrentHashMap(Blackhole bh) {
    for (int c = 0; c < size; c++) {
      bh.consume(chm.intern("doit" + c));
    }
  }
  public static class HashMapInterner {
    private final Map<String, String> map;
    public HashMapInterner() {
      map = new HashMap<>();
    }
    public String intern(String s) {
      String exist = map.putIfAbsent(s, s);
      return (exist == null) ? s : exist;
    }
  }
  @Benchmark
  public void useHashMap(Blackhole bh) {
    for (int c = 0; c < size; c++) {
      bh.consume(hm.intern("doit" + c));
    }
  }
  public static void main(String[] args) throws RunnerException {
    Options opt = new OptionsBuilder()
        .include(StringInternBenchMark.class.getSimpleName())
        .build();
    new Runner(opt).run();
  }
}

輸出結(jié)果:

Benchmark                                 (size)  Mode  Cnt          Score          Error  Units
StringInternBenchMark.useCurrentHashMap        1  avgt    5         34.259 ±        7.191  ns/op
StringInternBenchMark.useCurrentHashMap      100  avgt    5       3623.834 ±      499.806  ns/op
StringInternBenchMark.useCurrentHashMap    10000  avgt    5     421010.654 ±    53760.218  ns/op
StringInternBenchMark.useCurrentHashMap  1000000  avgt    5   88403817.753 ± 12719402.380  ns/op
StringInternBenchMark.useHashMap               1  avgt    5         36.927 ±        6.751  ns/op
StringInternBenchMark.useHashMap             100  avgt    5       3329.498 ±      595.923  ns/op
StringInternBenchMark.useHashMap           10000  avgt    5     417959.200 ±    62853.828  ns/op
StringInternBenchMark.useHashMap         1000000  avgt    5   79347127.709 ±  9378196.176  ns/op
StringInternBenchMark.useIntern                1  avgt    5        161.598 ±        9.128  ns/op
StringInternBenchMark.useIntern              100  avgt    5      17211.037 ±      188.929  ns/op
StringInternBenchMark.useIntern            10000  avgt    5    1934203.794 ±   272954.183  ns/op
StringInternBenchMark.useIntern          1000000  avgt    5  418729928.200 ± 86876278.365  ns/op

從結(jié)果我們可以看到,intern要比其他的兩個要慢。

所以native方法不一定快。intern的用處不是在于速度,而是在于節(jié)約Heap中的內(nèi)存使用。

到此這篇關(guān)于JVM系列之String.intern的性能解析的文章就介紹到這了,更多相關(guān)String.intern的性能內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 一起聊聊Java中的自定義異常

    一起聊聊Java中的自定義異常

    在學(xué)習(xí)Java的過程中,想必大家都一定學(xué)習(xí)過異常這個篇章,異常的基本特性和使用這里就不再多講了。本文就來和大家講講如何自定義異常
    2022-08-08
  • Java線程(Thread)四種停止方式代碼實例

    Java線程(Thread)四種停止方式代碼實例

    這篇文章主要介紹了Java線程(Thread)四種停止方式代碼實例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-03-03
  • Spring boot怎么整合Mybatis

    Spring boot怎么整合Mybatis

    spring boot的簡配置方便的開發(fā),下面通過本文給大家分享Spring boot整合Mybatis的方法,需要的朋友參考下
    2017-07-07
  • Spring 校驗(validator,JSR-303)簡單實現(xiàn)方式

    Spring 校驗(validator,JSR-303)簡單實現(xiàn)方式

    這篇文章主要介紹了Spring 校驗(validator,JSR-303)簡單實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Kotlin + Spring Boot 請求參數(shù)驗證的代碼實例

    Kotlin + Spring Boot 請求參數(shù)驗證的代碼實例

    本篇文章主要介紹了Kotlin + Spring Boot 請求參數(shù)驗證的代碼實例,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-07-07
  • java反射機制的一些學(xué)習(xí)心得小結(jié)

    java反射機制的一些學(xué)習(xí)心得小結(jié)

    這篇文章主要給大家介紹了關(guān)于java反射機制的一些學(xué)習(xí)心得,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • 修改SpringBoot 中MyBatis的mapper.xml文件位置的過程詳解

    修改SpringBoot 中MyBatis的mapper.xml文件位置的過程詳解

    由于MyBatis默認(rèn)的mapper.xml的掃描位置是resource文件下,但是不可能整個項目的mapper.xml文件都放在resource下,如果文件較少還行,但是如果文件比較多,太麻煩了,所以本文給大家介紹了修改SpringBoot 中MyBatis的mapper.xml文件位置的過程,需要的朋友可以參考下
    2024-08-08
  • java多線程編程之使用runnable接口創(chuàng)建線程

    java多線程編程之使用runnable接口創(chuàng)建線程

    實現(xiàn)Runnable接口的類必須使用Thread類的實例才能創(chuàng)建線程,通過Runnable接口創(chuàng)建線程分為以下兩步
    2014-01-01
  • 關(guān)于 Java 的數(shù)據(jù)結(jié)構(gòu)鏈表

    關(guān)于 Java 的數(shù)據(jù)結(jié)構(gòu)鏈表

    這篇文章主要介紹了關(guān)于 Java 的數(shù)據(jù)結(jié)構(gòu)鏈表的相關(guān)資料,需要的朋友可以參考下面文章內(nèi)容
    2021-09-09
  • Java實現(xiàn)五子棋游戲的完整代碼

    Java實現(xiàn)五子棋游戲的完整代碼

    這篇文章主要為大家詳細(xì)介紹了Java實現(xiàn)五子棋游戲的完整代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-10-10

最新評論