All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.esotericsoftware.kryo.benchmarks.ConcurrencyBenchmark Maven / Gradle / Ivy

There is a newer version: 5.6.0
Show newest version
/* Copyright (c) 2008-2023, Nathan Sweet
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following
 * conditions are met:
 *
 * - Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
 * - Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following
 * disclaimer in the documentation and/or other materials provided with the distribution.
 * - Neither the name of Esoteric Software nor the names of its contributors may be used to endorse or promote products derived
 * from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING,
 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
 * SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */

package com.esotericsoftware.kryo.benchmarks;

import com.esotericsoftware.kryo.Kryo;
import com.esotericsoftware.kryo.io.Input;
import com.esotericsoftware.kryo.io.Output;
import com.esotericsoftware.kryo.util.Pool;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.UUID;

import org.openjdk.jmh.annotations.Benchmark;
import org.openjdk.jmh.annotations.Scope;
import org.openjdk.jmh.annotations.State;
import org.openjdk.jmh.infra.Blackhole;
import org.openjdk.jmh.runner.Runner;
import org.openjdk.jmh.runner.RunnerException;
import org.openjdk.jmh.runner.options.Options;
import org.openjdk.jmh.runner.options.OptionsBuilder;
import org.openjdk.jmh.runner.options.TimeValue;

@State(Scope.Benchmark)
public class ConcurrencyBenchmark {

	@Benchmark
	public void single (SingleThreadState state, Blackhole blackhole) {
		final String data = UUID.randomUUID().toString();
		final Object result = state.roundTrip(data);
		blackhole.consume(result);
	}

	@Benchmark
	public void threadLocal (ThreadLocalState state, Blackhole blackhole) {
		final String data = UUID.randomUUID().toString();
		final Object result = state.roundTrip(data);
		blackhole.consume(result);
	}

	@Benchmark
	public void pool (PoolState state, Blackhole blackhole) {
		final String data = UUID.randomUUID().toString();
		final Object result = state.roundTrip(data);
		blackhole.consume(result);
	}

	@State(Scope.Benchmark)
	public static class SingleThreadState extends AbstractConcurrencyState {

		final Kryo kryo = createKryo();
		final Output output = createOutput();
		final Input input = createInput();

		public Object roundTrip (Object data) {
			synchronized (kryo) {
				final byte[] result = serialize(kryo, output, data);
				return deserialize(kryo, input, result);
			}
		}
	}

	@State(Scope.Benchmark)
	public static class ThreadLocalState extends AbstractConcurrencyState {

		static final ThreadLocal kryo = ThreadLocal.withInitial(ThreadLocalState::createKryo);
		static final ThreadLocal output = ThreadLocal.withInitial(ThreadLocalState::createOutput);
		static final ThreadLocal input = ThreadLocal.withInitial(ThreadLocalState::createInput);

		public Object roundTrip (Object data) {
			final Kryo k = kryo.get();
			final Output out = output.get();
			final Input in = input.get();
			final byte[] result = serialize(k, out, data);
			return deserialize(k, in, result);
		}
	}

	@State(Scope.Benchmark)
	public static class PoolState extends AbstractConcurrencyState {

		static final int CAPACITY = 8;

		static final Pool kryoPool = new Pool(true, false, CAPACITY) {
			protected Kryo create () {
				return createKryo();
			}
		};
		static final Pool outputPool = new Pool(true, false, CAPACITY) {
			protected Output create () {
				return createOutput();
			}
		};
		static final Pool inputPool = new Pool(true, false, CAPACITY) {
			protected Input create () {
				return createInput();
			}
		};

		public Object roundTrip (Object data) {
			final Kryo k = kryoPool.obtain();
			final Output out = outputPool.obtain();
			final Input in = inputPool.obtain();
			try {
				final byte[] result = serialize(k, out, data);
				outputPool.free(out);
				return deserialize(k, in, result);
			} finally {
				kryoPool.free(k);
				inputPool.free(in);
			}
		}
	}

	public abstract static class AbstractConcurrencyState {

		public abstract Object roundTrip (Object data);

		byte[] serialize (Kryo kryo, Output output, Object data) {
			try (final ByteArrayOutputStream stream = new ByteArrayOutputStream()) {
				output.setOutputStream(stream);
				kryo.writeClassAndObject(output, data);
				output.flush();
				return stream.toByteArray();
			} catch (IOException e) {
				throw new RuntimeException(e);
			} finally {
				output.reset();
			}
		}

		Object deserialize (Kryo kryo, Input input, byte[] data) {
			try {
				input.setBuffer(data);
				return kryo.readClassAndObject(input);
			} finally {
				input.reset();
			}
		}

		static Kryo createKryo () {
			final Kryo kryo = new Kryo();
			kryo.register(ArrayList.class);
			return kryo;
		}

		static Output createOutput () {
			return new Output(4096, -1);
		}

		static Input createInput () {
			return new Input(4096);
		}
	}

	public static void main (String[] args) throws RunnerException {
		final Options opt = new OptionsBuilder()
			.include(".*" + ConcurrencyBenchmark.class.getSimpleName() + ".*")
			.warmupIterations(3)
			.warmupTime(TimeValue.seconds(3))
			.measurementIterations(3)
			.measurementTime(TimeValue.seconds(3))
			.threads(2)
			.forks(1)
			.build();
		new Runner(opt).run();

	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy