Java is Very Fast, If You Don’t Create Many Objects

 

You still have to watch how many objects you create.


This article looks at a benchmark passing events over TCP/IP at 4 billion events per minute using the net.openhft.chronicle.wire.channel package in Chronicle Wire and why we still avoid object allocations.. 

One of the key optimisations is creating almost no garbage. Allocation is a very cheap operation and collection of very short-lived objects is also very cheap. Does this really make a difference? What difference does one small object per event (44 bytes) make to the performance in a throughput test where GC pauses are amortised?

While allocation is as efficient as possible, it doesn’t avoid the memory pressure on the L1/L2 caches of your CPUs and when many cores are busy, they are contending for memory in the shared L3 cache. 

Results

Benchmark on a Ryzen 5950X with Ubuntu 22.10.

JVM Vendor, Version

No objects

Throughput, Average Latency*

One object per event

Throughput, Average Latency*

Azul Zulu 1.8.0_322

60.6 M event/s, 528 ns

36.8 M event/s, 879 ns

Azul Zulu 11.0.14.1

67.3 M event/s, 476 ns

45.7 M event/s, 700 ns

Azul Zulu 17.0.4.1

68.6 M event/s, 467 ns

50.3 M event/s, 636 ns

Azul Zulu 18.0.2.1

67.5 M event/s, 474 ns

49.8 M event/s, 642 ns

Oracle OpenJDK 18.0.2.1

67.8 M event/s, 472 ns

50.1 M event/s, 638 ns

* Across 16 clients, an event is sent in both directions. The Average Latency = 2 * 16 / throughput


One extra object for each event adds 166 ns or so.  This doesn’t sound like much; however, in a high throughput context, this can reduce performance by 25%.

The default behaviour for reading events in Chronicle WIre is to reuse the same object for the same event type every time on deserialization. This provides a simple object pooling strategy to avoid allocations. If this data is retained across method calls, it must be copied.

NOTE: The total time spent in GC was about 170 milliseconds per minute or 0.3% of the time. It is the allocations rather than the time to clean up these very short-lived objects that takes time.


A benchmark of creating short-lived TopOfBook objects across multiple CPUs, produces a similar result. This suggests that the rate new objects can be allocated is quickly saturated for even a small proportion of cores,  increasing the average latency with more threads. This is for the same small 44-byte objects.


On a Ryzen 5950X with Ubuntu 21.10, Java 17.0.4.1

Threads

Allocation rate, average latency

2

126 M objs/sec, 15 ns

3

176 M objs/sec, 17 ns

4

211 M objs/sec, 18 ns

6

226 M objs/sec, 26 ns

8

219 M objs/sec, 36 ns

16

220 M objs/sec, 72 ns

32

211 M objs/sec, 150 ns

32 (two JVMS)

209 M objs/sec total

The benchmarks

They are not directly comparable but confirm that it’s in the same ballpark. See PerfTopOfBookAllocationMain 

In this benchmark, sixteen clients connect to a simple microservice that takes each event and sends it back again. All events are (de)serialised POJOs with an event type. This translates to an asynchronous RPC call.


public class EchoTopOfBookHandler implements ITopOfBookHandler {

   private TopOfBookListener topOfBookListener;


   @Override

   public void topOfBook(TopOfBook topOfBook) {

       if (ONE__NEW_OBJECT)

           topOfBook = topOfBook.deepCopy();

       topOfBookListener.topOfBook(topOfBook);

   }


In this case, deepCopy() creates a new TopOfBook and sets all the fields.


The benchmark can be run in two modes, one where no objects are allocated and one where any object is allocated and initialised, allowing us to measure the difference this makes. Each event is modelled as an asynchronous RPC call to make testing, development and maintenance easier. 


public interface TopOfBookListener {

   void topOfBook(TopOfBook topOfBook);

}


Low latency software can be very fast but also difficult to work with, slowing development. Using events modelled in YAML we can support Behaviour Driven Development of the microservice.


This is the in.yaml and the out.yaml for the microservice above.


# first top-of-book

---

topOfBook: {

 sendingTimeNS: 2022-09-05T12:34:56.789012345,

 symbol: EUR/USD,

 ecn: EBS,

 bidPrice: 0.9913,

 askPrice: 0.9917,

 bidQuantity: 1000000,

 askQuantity: 2500000

}

...

# second top-of-book

---

topOfBook: {

 sendingTimeNS: 2022-09-05T12:34:56.789123456,

 symbol: EUR/USD,

 ecn: EBS,

 bidPrice: 0.9914,

 askPrice: 0.9918,

 bidQuantity: 1500000,

 askQuantity: 2000000

}

...

The code

Is available here https://github.com/OpenHFT/Chronicle-Wire/tree/ea/src/test/java/net/openhft/chronicle/wire/channel/book

This library is used in Chronicle Services

Hacker News comments on this article https://news.ycombinator.com/item?id=32807587#32809487

Twitter post on this article https://twitter.com/PeterLawrey/status/1569247849074012160

Conclusion

Java can be very fast, however, it can be well worth avoiding object creation. 

The cost of object creation can be far higher than cleaning them up if they are very short-lived.


Comments

  1. About your closing comment, how does the benchmark distinguish added creation time vs GC time? Every GC (except Epsilon) has some stop-the-world portion, even ZGC needs to for root marking. I would have thought that this would mean GC always entails some risk of losing a core. And therefore I had assumed that GC would add more delays than allocation.

    Could you please clarify that point, or point me to a post where you do?

    ReplyDelete
    Replies
    1. The GC adds bigger delays, and has an impact of the higher percentiles of latency. However in a throughput test it doesn't add up to much. In the benchmark above, it spent 170 ms every minute in a GC. This stops the application 0.3% of the time. By comparison the reduction in throughput due to the allocations slows the throughput by 25%

      Delete

Post a Comment

Popular posts from this blog

System wide unique nanosecond timestamps

Comparing Approaches to Durability in Low Latency Messaging Queues