Java Platform Evolution
Java has been around since 1995 and is used by over 10 million developers in almost every country in the world. It truly is one of the most successful technologies in history. But this doesn't mean the platform is standing still. Quite the opposite in fact.
Java 8, released in 2014, brought Lambda expressions to the Java platform as well as the Stream API, Optional class, and a host of other great features. This is one of the reasons why Java 8 became the most popular version in Java's history. Even today, it remains one of the most widely used versions of Java.
But choosing Java 8 today prevents developers from gaining access to an immense amount of progress in the language, JVM, tooling, and more. This article will attempt to summarize much of this progress. But don't just take our word for it, go try the latest version of Java today!
Thoughtful Evolution
Before we dive in, let's take a minute to view how the stewards of Java think about the evolution of the Java Platform. Brian Goetz, Java Language Architect, talks quite a bit about this subject. One talk in particular, Stewardship: The Sobering Parts, is a great starting point. Brian talks about the challenges, and joys, of being that unique type of person who is happy stewarding a language over many years, carefully balancing the forces of conservatism (move slower, stay compatible) with innovation (move faster, adapt to change), only to continually find out you've upset people on both sides.
Learn more on our stewardship page with talks from Mark Reinhold and John Rose.
Accelerating Innovation
Now with this thoughtful evolution in mind and the careful balance of conservatism and innovation, the Java team set out to increase innovation. The foundation of this effort was in changes to Java's long-standing release cadence that was feature driven over many year cycles. First proposed by Mark Reinhold in his Moving Java Forward Faster post, Java moved to a 6-month, time-based release cadence in 2017 with a new feature release of Java becoming available like clockwork every March and September. This shift has had a dramatic impact on the ecosystem for many reasons, some obvious, and some not.
The obvious benefit to the community has been the ability for developers to get their hands on new Java features every 6 months as opposed to every 2-3 years. Multiple years is a long time to wait, providing many opportunities for developers to search elsewhere for a different tool to solve their problem. This cycle contributed to the perception that Java was "standing still" even though it wasn't. Consider this: after Java 9 in 2017, the next release under the old cadence would likely have happened in 2020. A lot can happen in three years -- and in Java, it did! For example, under the new cadence, we delivered var in Java 10 in 2018, and made significant improvements to the G1 garbage collector in Java 12 in 2019. With a wide variety of new features in each six-month release, there's no reason to miss out by only looking at Java every three years.
In addition, with 6-month cycles, the team has been able to effectively introduce (or improve upon) models to introduce features such as the preview system, and time-based early access builds, generating feedback from the community before features become standard, at which point they will be around for a very long time.
What's more, the release cadence has allowed teams to break up large features (Project Loom, Panama, etc.) and introduce them, over time, as incremental changes resulting in better project planning and better end-user experiences. New versions of Java can be tested and introduced automatically using CI/CD pipelines, rather than big scary stop-the-world events, and users will have plenty of lead time to get used to new features. For an example of this, see the section on project Amber below.
Finally, a not-so-obvious benefit is a story of productivity and happiness for the teams working on the Java platform itself. Releases are mostly "non-events" now as they've been solidified, ready, and in many cases tested in the wild through our preview and early-access models for weeks and months in advance.
Snapshot Since 8
For more maintainable programs, check out Records, Sealed Classes, and Pattern Matching.
For more concise programs, check out JEP 395: Records and JEP 463: Implicitly Declared Classes and Instance Main Methods (Second Preview).
For pain-free multi-line strings, check out Text Blocks.
For reducing the effort of writing, maintaining, and observing high-throughput concurrent applications, check out Virtual Threads.
For better documentation, check out JEP 225: Javadoc Search and JEP 413: Code Snippets in Java API Documentation.
For experimenting and learning, check out jshell, JEP 408: Simple Web Server, and the Java Playground.
For easier debugging, check out JEP 358: Helpful NullPointerExceptions and JEP 349: JFR Event Streaming.
And for the whole laundry list of new stuff since 8, keep scrolling to the laundry list section.
Project Amber in Action
A picture is worth a thousand words. Here is a picture that represents Project Amber, the release cadence, and the preview system in action over the last few years. As you can see, many of Amber's language features started off as preview features to collect broader feedback, and sometimes made slight changes before becoming standard.
Learn more about Project Amber at its wiki and Inside.java.
Learn more about the Preview system here.
Performance
One of the most important evolutions of the Java platform over the years has been in performance. Everything from "out of the box" performance by just upgrading, to taking advantage of modern hardware, to consistent pause times with very large heaps, the Java Platform is getting better and better with each release.
With the help of the Java GC team, here are a few snapshots of the performance evolution over time, pulled from a blog post by Stefan Johansson. To find more great information on performance directly from the Java team, check out Inside.java.
All results are from running the SPECjbb 2015 benchmarks with a fixed heap size of 16GB, and no tuning of GC parameters. First, you can see that the throughput achieved by each collector is higher in Java 17 than in previous releases. ZGC reached production status with Java 15.
These are average pause times, normalized for each collector in turn. The average pause time for Parallel in 17 is approximately 60% of its average pause time in 8. Similarly for G1.
That's a pretty good reason to try your application on 17. But of course, your eye is drawn to the ZGC area. ZGC in 17 has way lower pause times than it had in 15.
This is a LOG graph of the ACTUAL average pause times, but with an additional column for 128GB heap. Because ZGC is fully concurrent, it achieves sub-millisecond pause times on the same workload as other collectors.
Notice that Parallel and G1 have improved so much between 8 and 17 that you may be able to octuple the size of your data and still run with lower pauses in 17 than in 8. Let that sink in. You may be able to octuple the size of your data and still run with lower pause times in 17, with no changes to your code, no special tuning, no third-party tools. Just the Java you know and love.
You'll also notice that ZGC's pause times are constant. You can tune G1 and Parallel to have shorter pauses, but it will be hard to reach ZGC. When your product faces a 10x increase in the amount of data it has to process, the best response is to move to Java 17 and utilize G1 and ZGC.
Finally let's consider footprint. Footprint means the overhead of GC data structures in native memory, so the Java heap can't use the memory. A GC using less native memory lets you co-locate more JVMs on the same machine.
Back in Java 8, before G1 became the default, it usually had an overhead of around 20%, and that's down to around 10% in Java 17. This benchmark is relatively kind: it shows G1's overhead around 10% on Java 8, down to around 5% on Java 17.
That's roughly the same as the overhead of the Parallel collector, so moving from 8 to 17 and getting G1 by default shouldn't be significant memory-wise.
We see that ZGC's low latency does come with some cost of a higher footprint, but that'll improve over releases.
Laundry List Since 8
For those that want a larger laundry list of improvements to the Java platform since Java 8, categorized for easier scanning.
Language Features
- JEP 286: Local-Variable Type Inference
- Switch Expressions
- Records
- Sealed Classes
- Text Blocks
- Pattern Matching for instanceof
- Pattern Matching for Switch
Library and Tooling
- JEP 408: Simple Web Server
- JEP 413: Code Snippets in Java API Documentation
- The Java Shell Tool
- Foreign Function & Memory API
Security
- Strongly Encapsulate JDK Internals
- Enhanced Pseudo-Random Number Generators
- JEP 415: Context-Specific Deserialization Filters
Observability and Debugging
Garbage Collection
- ZGC: A Scalable Low-Latency Garbage Collector
- ZGC: Uncommit Unused Memory
- JEP 345: NUMA-Aware Memory Allocation for G1
- JEP 346: Promptly Return Unused Committed Memory from G1
- JEP 376: ZGC: Concurrent Thread-Stack Processing
- JEP 387: Elastic Metaspace
Modernizing Infrastructure
Removals and Deprecations
- JEP 363: Remove the Concurrent Mark Sweep (CMS) Garbage Collector
- JEP 372: Remove the Nashorn JavaScript Engine
- JEP 374: Disable and Deprecate Biased Locking
- JEP 398: Deprecate the Applet for Removal
- JEP 407: Remove RMI Activation
- JEP 411: Deprecate the Security Manager for Removal
- JEP 421: Deprecate Finalization for Removal
Misc
- JEP 400: UTF-8 by Default
- JEP 416: Reimplement Core Reflection with Method Handle
- JEP 418: Internet-Address Resolution API
Bonus Section: Preview/Incubator Features
More Learning
Last update: November 14, 2023
Back to Tutorial List