Tag Archives: Performance

Performance — performance optimization, benchmarking, and profiling in Java applications

Mastering Hibernate 6 L2 Caching with Ehcache 3: The Definitive Guide

Hibernate 6 introduced a major architectural shift. By moving to the Jakarta EE namespace and embracing the JCache (JSR-107) standard, it changed how we interact with caching providers. If you are using Hibernate 6, the legacy hibernate-ehcache dependency is dead. To achieve high-performance data access today, you need the modern JCache bridge.

The Problem: Database Bottlenecks in Hibernate 6

Even with the performance improvements in Hibernate 6’s new SQM (Semantic Query Model), database latency remains the primary bottleneck for scaling. Without a Second Level (L2) Cache, every time a new Session (EntityManager) is opened—even for the same user—Hibernate is forced to hit the database for data that hasn’t changed.

This results in redundant SQL SELECT statements, higher DB CPU usage, and increased costs in cloud environments where you pay for IOPS and database instance sizing.

The Solution: Hibernate 6 + Ehcache 3 (JSR-107)

The modern solution for Hibernate 6 is to use Ehcache 3 as a JCache provider. This allows Hibernate to offload entity and collection state to memory, sharing it across all sessions in the SessionFactory.

Prerequisites

  • Java 17+: Hibernate 6 requires a minimum of Java 17.
  • Jakarta Persistence 3.x: The modern jakarta.persistence namespace.

Step 1: Hibernate 6 Dependencies

In Hibernate 6, you must use the hibernate-jcache module. Crucially, your Ehcache dependency must include the jakarta classifier to avoid namespace conflicts.

Continue reading Mastering Hibernate 6 L2 Caching with Ehcache 3: The Definitive Guide

Master Hibernate 7 Ehcache 3 Configuration: High-Performance Caching with Jakarta Persistence

Are you struggling with database performance in your modern Java applications? As systems scale, database bottlenecks remain the primary cause of latency. While Hibernate 7 introduces massive improvements in query generation and Jakarta Persistence compatibility, the secret to true high-concurrency performance lies in the Second-Level (L2) Cache. In this guide, we will configure Ehcache 3 — the industry standard for JVM-level caching — as the L2 cache provider for Hibernate 7, using the modern JCache (JSR-107) bridge.

Continue reading Master Hibernate 7 Ehcache 3 Configuration: High-Performance Caching with Jakarta Persistence

Hibernate 7 Second-Level Cache: When to Turn It On, How to Configure Ehcache 3, and the Three Ways It Will Stale Your Data

The question that matters before enabling L2 cache is not “can I make it faster” but “can I tolerate stale data, and for how long”. The cache sits between Hibernate and the database, serving data that may have been written by a different JVM process, a batch job, a DBA running a script, or another application instance. Every one of those paths can invalidate the cache without Hibernate knowing. If your answer to the staleness question is “no, I cannot tolerate any staleness” — skip L2 entirely. If the answer is “yes, but only for these entities and within these bounds” — read on.

This post covers what L2 actually stores, how to configure Ehcache 3 as a JCache provider, what changed between Hibernate 6 and 7, the three specific staleness modes that catch production teams off guard, and why the query cache is almost always the wrong additional layer.

Continue reading Hibernate 7 Second-Level Cache: When to Turn It On, How to Configure Ehcache 3, and the Three Ways It Will Stale Your Data

The Hibernate First-Level Cache Explained (It’s Not What You Think It Is)

Most developers who have used Hibernate for any length of time know the first-level cache exists. Ask them to describe it and you’ll hear something like: “It’s a cache Hibernate uses so it doesn’t hit the database twice for the same row.” That’s technically correct, but it misses almost everything that matters.

The first-level cache is not a feature you enable, a setting you tune, or an optional layer you bolt on for performance. It is the persistence context itself — always-on, transaction-scoped by default in Spring, and the thing that makes dirty checking, identity guarantees, and cascade operations possible. If you have ever used em.find(), you have used it. If you have ever hit an OutOfMemoryError in a batch job that loaded 200,000 entities, the first-level cache is why.

This post is a deep look at the mechanics: what the persistence context stores, when it is consulted, how EntityKey works, and the failure modes that catch experienced developers off guard.

Continue reading The Hibernate First-Level Cache Explained (It’s Not What You Think It Is)

Mastering Stored Procedures with Hibernate 7: A Deep Dive for High-Performance Java Apps

If you’ve ever written nested loops in Java just to process thousands of records—only to watch latency skyrocket—you’re not alone. This application–database “chattiness” is a silent performance killer that creeps into enterprise systems as they scale. Every time your application fetches a row, transforms it in memory, and sends it back to the server, you incur the cumulative overhead of network round-trips, intensive object-relational mapping (ORM) overhead, and heavy JVM garbage collection cycles. For a few dozen records, this is negligible; for a few million, it is a catastrophic bottleneck that can bring a production environment to its knees.

The solution? Shift data‑intensive logic into the database layer and invoke it through Hibernate 7. In this guide, you’ll learn when and how to use stored procedures safely, portably, and efficiently.


Why Use Stored Procedures in Hibernate 7?

Hibernate 7 continues to improve support for native database features while aligning fully with Jakarta Persistence 3.2. Stored procedures are not a silver bullet, but in the right scenarios they offer tangible benefits:

  • Reduce Network Latency: Execute complex logic in a single round-trip instead of hundreds of individual queries.
  • Centralize Logic: Keep data-heavy calculations close to the data to avoid serialization overhead.
  • Security: Expose only procedures instead of granting direct table access.
  • Traffic Optimization: Offload bulk relational work to the database engine, which is optimized for it.

Note: Hibernate also provides the native ProcedureCall API for finer-grained control. For most portable applications, prefer the JPA-standard StoredProcedureQuery.


The PAS Framework: Problem, Agitation, Solution

The Problem

Your application must calculate a year-end bonus based on multiple tables: Sales, Attendance, Tenure, and departmental performance.

Continue reading Mastering Stored Procedures with Hibernate 7: A Deep Dive for High-Performance Java Apps

BLOB and CLOB in Hibernate 7: Streaming vs Eager, and the OOM You Didn’t See Coming

A list endpoint returned a page of 100 products. Simple enough — a cheap SELECT should be fast. But the page timed out, the heap spiked to 4 GB, and the GC ran continuously. The cause: the Product entity had an @Lob byte[] thumbnail field mapped with default eager fetching. Each of the 100 products loaded its thumbnail — averaging 40 MB each — all at once, into the heap. 100 rows × 40 MB = 4 GB from a list query that didn’t display thumbnails.

This is the OOM you don’t see coming because the entity mapping looks harmless. This post covers the difference between byte[] (always eager), Blob with bytecode enhancement (genuinely lazy), and streaming (no heap allocation at all) — with approximate memory numbers for each.

If you are building modern Java applications, handling BLOB and CLOB with Hibernate 7 is a skill you cannot ignore. In this guide, we will dive deep into how to efficiently map, persist, and retrieve binary and character data using the latest Hibernate standards aligned with Jakarta Persistence 3.2+.


The Problem: The “Out of Memory” Nightmare

Storing small strings like usernames or emails is easy. But what happens when your data grows to megabytes? Traditional mapping techniques often try to load the entire object into the JVM’s memory.

Imagine a scenario where 100 concurrent users try to download a 50MB PDF. If your application is configured to load the entire file into a byte[], your server will attempt to allocate 5GB of RAM instantly. In most environments, this leads to the dreaded:

java.lang.OutOfMemoryError: Java heap space

This crashes your service and disrupts all users.

Continue reading BLOB and CLOB in Hibernate 7: Streaming vs Eager, and the OOM You Didn’t See Coming

@OneToMany Done Right: Set vs List, Bidirectional Sync, and the MultipleBagFetchException Trap

Hibernate has a strong preference between Set<Child> and List<Child> in a @OneToMany mapping. If you use List and try to JOIN FETCH two collections simultaneously, you get MultipleBagFetchException. If you use List and remove one child, Hibernate deletes all children and reinserts the remainder. If you use Set, a single-child removal fires one targeted DELETE. The performance difference on a parent with 500 children is the difference between 1 SQL statement and 501.

This post covers Set vs List semantics, why the bidirectional sync helper method matters and what breaks without it, and the MultipleBagFetchException with its three fixes.

The Problem: Data Fragmentation and Manual Syncing

Imagine you are building an e-commerce platform. A single Customer can place multiple Orders. In a raw SQL world, you’d have to manually manage foreign keys, write complex joins, and ensure that when a customer is deleted, their orphaned orders don’t break your database integrity.

Manually mapping these relationships in Java code leads to “Boilerplate Hell”—hundreds of lines of code spent manually updating IDs, checking for nulls, and keeping two separate objects in sync. This manual labor is error-prone and often leads to data inconsistency between your application memory and the actual database state.

Continue reading @OneToMany Done Right: Set vs List, Bidirectional Sync, and the MultipleBagFetchException Trap