3

+演算子とStringBuilderを使用して文字列連結を比較するためのマイクロベンチマークを作成しています。この目的のために、 batchSizeパラメータを使用する OpenJDK の例に基づいて JMH ベンチマーク クラスを作成しました。

@State(Scope.Thread)
@BenchmarkMode(Mode.AverageTime)
@Measurement(batchSize = 10000, iterations = 10)
@Warmup(batchSize = 10000, iterations = 10)
@Fork(1)
public class StringConcatenationBenchmark {

    private String string;

    private StringBuilder stringBuilder;

    @Setup(Level.Iteration)
    public void setup() {
        string = "";
        stringBuilder = new StringBuilder();
    }

    @Benchmark
    public void stringConcatenation() {
        string += "some more data";
    }

    @Benchmark
    public void stringBuilderConcatenation() {
        stringBuilder.append("some more data");
    }

}

ベンチマークを実行すると、stringBuilderConcatenationメソッドに対して次のエラーが発生します。

java.lang.OutOfMemoryError: Java heap space
    at java.util.Arrays.copyOf(Arrays.java:3332)
    at java.lang.AbstractStringBuilder.expandCapacity(AbstractStringBuilder.java:137)
    at java.lang.AbstractStringBuilder.ensureCapacityInternal(AbstractStringBuilder.java:121)
    at java.lang.AbstractStringBuilder.append(AbstractStringBuilder.java:421)
    at java.lang.StringBuilder.append(StringBuilder.java:136)
    at link.pellegrino.string_concatenation.StringConcatenationBenchmark.stringBuilderConcatenation(StringConcatenationBenchmark.java:29)
    at link.pellegrino.string_concatenation.generated.StringConcatenationBenchmark_stringBuilderConcatenation.stringBuilderConcatenation_avgt_jmhStub(StringConcatenationBenchmark_stringBuilderConcatenation.java:165)
    at link.pellegrino.string_concatenation.generated.StringConcatenationBenchmark_stringBuilderConcatenation.stringBuilderConcatenation_AverageTime(StringConcatenationBenchmark_stringBuilderConcatenation.java:130)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:497)
    at org.openjdk.jmh.runner.BenchmarkHandler$BenchmarkTask.call(BenchmarkHandler.java:430)
    at org.openjdk.jmh.runner.BenchmarkHandler$BenchmarkTask.call(BenchmarkHandler.java:412)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at java.lang.Thread.run(Thread.java:745)

デフォルトの JVM ヒープ サイズを増やす必要があると考えていたので、JMH が提供するオプション-Xmx10Gで値を使用して最大 10GB まで許可しようとしました。-jvmArgs残念ながら、まだエラーが発生します。

batchSizeその結果、パラメーターの値を1に減らそうとしましたが、それでもOutOfMemoryErrorが発生します。

私が見つけた唯一の回避策は、ベンチマーク モードを に設定することMode.SingleShotTimeです。このモードではバッチを 1 回のショットと見なしているように見えるため (単位列にs/opが表示されていても)、必要なメトリック、つまり一連のバッチ操作を実行する平均時間を取得しているようです。ただし、なぜそれが動作しないのかはまだわかりませんMode.AverageTime

stringConcatenationメソッドのベンチマークは、どのベンチマーク モードを使用しても期待どおりに機能することにも注意してください。stringBuilderConcatenationこの問題は、StringBuilder を使用するメソッドでのみ発生します。

ベンチマーク モードを に設定した場合に前の例が機能しない理由を理解するための助けをMode.AverageTime歓迎します。

私が使用した JMH のバージョンは1.10.4です。

4

1 に答える 1