Demystifying Java Object Sizes: Compact Headers, Compressed Oops, and Beyond

Introduction

Measuring an object’s size in Java is not straightforward. The platform encourages you to consider references and abstractions rather than raw memory usage. Still, understanding how objects fit into memory can yield significant benefits, especially for high-performance, low-latency systems.

Over time, the JVM has introduced optimisations like Compressed Ordinary Object Pointers (Compressed Oops) and, more recently, Compact Object Headers. Each of these can influence how large or small your objects appear. Understanding these factors helps you reason about memory usage more concretely.

Measuring Object Sizes

In principle, you can estimate an object’s size by creating instances and observing changes in the JVM’s free memory. However, you must neutralise certain factors to get consistent results. For example, turning off TLAB allocation (-XX:-UseTLAB) makes memory usage more directly observable. Repeated measurements and median calculations can reduce the impact of GC and concurrent allocations.

A GC can occur while you are creating your object. This will result in more free memory at the end than when you started. I ignore any negative sizes in this test ;) Other threads in the system could use memory at the same time. I perform multiple test and take the median, which removes any outliers.

Below is a rough approach:

long before = usedMemory();
Object obj = createYourObject();
long after = usedMemory();
long approximateSize = after - before;

This test SizeofTest.java is a simple test which creates a number of objects and measures the memory used to create each object. This is usually the same as the amount of memory the object retains for simple objects.

Approximate layout of an object

Memory Region Description Size (Bytes)

Mark Word

Header information including identity hash code, lock state, and GC metadata

8 bytes (on 64-bit JVMs)

Class Pointer (Klass Pointer)

Reference to the object’s class metadata, used internally by the JVM

Typically 4 bytes with Compressed Class Pointers, otherwise 8 bytes

Array Length (Arrays Only)

Stores the length of the array; present only for array objects

4 bytes

Instance Fields

Actual data fields of the object: primitives and references

Varies depending on field types and alignment

Padding

Unused space to ensure proper 8-byte alignment of the object size

0 to 7 bytes, as needed

JEP 450: Compact Object Headers (Experimental)

Starting with Java 24 (Early Access), the JVM introduces Compact Object Headers via Java Enhancement Proposal 450

Summary

Reduce the size of object headers in the HotSpot JVM from between 96 and 128 bits down to 64 bits on 64-bit architectures. This will reduce heap size, improve deployment density, and increase data locality.

Goals

When enabled, this feature

  • Must reduce the object header size to 64 bits (8 bytes) on the target 64-bit platforms (x64 and AArch64),

  • Should reduce object sizes and memory footprint on realistic workloads,

  • Should not introduce more than 5% throughput or latency overheads on the target 64-bit platforms, and only in infrequent cases, and

  • Should not introduce measurable throughput or latency overheads on non-target 64-bit platforms.

Performance Considerations

A smaller object footprint can improve performance by increasing the number of objects that fit into CPU caches, reducing the cost of GC, and even improving startup times. However, enabling these features (or disabling them) should be accompanied by testing and benchmarking (e.g., using JMH). Depending on your workload, these optimisations may yield modest gains or have negligible impact.

These improvements can translate into substantial cost savings and better utilisation of available resources for memory-sensitive environments, especially those running microservices where each instance might have limited heap space.

Sizeof objects with different options

  • Compact Headers: -XX:+UnlockExperimentalVMOptions -XX:+UseCompactObjectHeaders in Java 24 EA

  • Compressed Oops: -XX:+UseCompressedOops added in Java 8 as the default for a heap size up to 32 GB

  • No Compressed Oops: -XX:-UseCompressedOops for heap sizes over 32 GB

Table 1. Wrappers
Class/Object Compact Header +Compressed Oops -Compressed Oops

Object

8

16

16

Boolean, Bytes

16

16

16

Short, Character

16

16

16

Integer, Float

16

16

16

Long, Double

16

24

24

Table 2. Simple CLasses
Class/Object Compact Header +Compressed Oops -Compressed Oops

AtomicInteger

16

16

16

AtomicReference

16

16

24

AtomicLong

16

24

24

Optional, SimpleEntry

16

16

24

"Hello World"

24

24

32

CompletableFuture

16

24

32

WeakReference

32

48

64

StringBuilder

56

56

64

Pattern

1056

1088

1240

UUID

216

240

256

Exception

712

728

896

Locale

80

104

120

Table 3. Date and Time
Class/Object Compact Header +Compressed Oops -Compressed Oops

Date

24

24

32

Timestamp

24

32

32

TimeZone

56

56

80

LocalDate, LocalTime

128

136

168

LocalDateTime

160

184

224

ZonedDateTime

208

232

288

Calendar

528

560

648

Instant, Duration, Period

24

24

24

ZoneId

56

56

80

Table 4. Collections with zero / one / ten elements
Class/Object Compact Header +Compressed Oops -Compressed Oops

ArrayList

24 / 80 / 80

24 / 80 / 80

32 / 128 / 128

LinkedList

24 / 48 / 264

32 / 56 / 272

40 / 80 / 440

ConcurrentLinkedQueue

32 / 48 / 192

48 / 72 / 288

64 / 96 / 384

ConcurrentHashMap

64 / 168 / 384

64 / 176 / 464

96 / 280 / 384

TreeMap

48 / 80 / 368

48 / 88 / 448

80 / 136 / 640

TreeSet

64 / 96 / 384

64 / 160 / 464

104 / 160 / 664

HashMap

40 / 144 / 360

48 / 160 / 448

64 / 248 / 608

HashSet

56 / 160 / 376

64 / 176 / 464

88 / 272 / 632

LinkedHashMap

56 / 168 / 456

64 / 184 / 544

88 / 288 / 792

LinkedHashSet

72 / 184 / 472

80 / 200 / 560

112 / 312 / 816

Vector, Stack

80 / 80 / 80

88 / 88 / 88

128 / 128 / 128

Hashtable

96 / 120 / 440

112 / 144 / 544

168 / 208 / 768

Table 5. Arrays
Class/Object Compact Header +Compressed Oops -Compressed Oops

new BitSet(64)

48

48

56

new boolean[64], new byte[64]

80

80

80

new short[64], new char[64]

144

144

144

new int[64], new float[64]

272

272

272

new long[64], new double[64]

528

528

528

new Object[64], new Integer[64], new String[64], new Long[64], new Double[64]

272

272

528

These values are approximate, environment-dependent, and should be considered illustrative rather than absolute.

About the author

As the CEO of Chronicle Software, Peter Lawrey leads the development of cutting-edge, low-latency solutions trusted by 8 out of the top 11 global investment banks. With decades of experience in the financial technology sector, he specialises in delivering ultra-efficient enabling technology which empowers businesses to handle massive volumes of data with unparalleled speed and reliability. Peter’s deep technical expertise and passion for sharing knowledge have established him as a thought leader and mentor in the Java and FinTech communities. Follow Peter on BlueSky or Mastodon.

Conclusion

Java’s abstractions often free you from worrying about memory details, but understanding how object sizes change with different JVM configurations can help fine-tune performance. Compact Headers and Compressed Oops are powerful features that let you reduce memory footprints and potentially improve efficiency. Experimentation, measurements, and thoughtful benchmarking will guide you in making informed choices for your specific workloads.

In my experience, scrutinising object sizes has proven valuable when dealing with large-scale, memory-sensitive applications. Testing these configurations allows you to discover the right balance for your services and potentially save on infrastructure costs.

Consider starting small: measure a few objects, toggle TLAB or Compressed Oops settings, and see what changes. Over time, you will build a deeper mental model of how Java’s memory behaves, enabling you to write more efficient and predictable code.

Comments

  1. Thank you to Ben Evans for noting the size is much higher for collections with even one element

    ReplyDelete

Post a Comment

Popular posts from this blog

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

System wide unique nanosecond timestamps

Comparing Approaches to Durability in Low Latency Messaging Queues