Reference:
[A categorized list of all Java and JVM features since JDK 8 to 18](A categorized list of all Java and JVM features since JDK 8 to 18 - Advanced Web Machinery)
full list of JDK Enhancement Proposals: OpenJDK "JEPs in JDK 17 integrated since JDK 11".
Highlights for JDK from 11 to 17
Text Blocks
Example:
// BEFOREString oldBlock = "This is the old way\n" + "of doing multi-line\n" + "strings.";// AFTERString newBlock = """ This is the new way of doing multi-line strings. """;
Switch Expressions
For an in-depth example, see entry on Switch Expressions.
With switch expressions support, developers can now streamline their switch blocks by omitting the break
statements.
private static void withReturnValueEvenShorter(Fruit fruit) { System.out.println( switch (fruit) { case APPLE, PEAR -> "Common fruit"; case ORANGE, AVOCADO -> "Exotic fruit"; default -> "Undefined fruit"; });}
Records
For an in-depth example, see entry on Records.
Records will allow you to create immutable data classes. Currently you need to create a class using the autogenerate functions of your IDE to generate constructor, getters, hashCode
, equals
and toString
you need to write your own getter, setter, or use Lombok for this purpose.
// BEFOREpublic class Article { private String title; private String content; private Long authorId; public String getTitle() { return title; } public void setTitle(String title) { this.title = title; } public String getContent() { return content; } public void setContent(String content) { this.content = content; } public Long getAuthorId() { return authorId; } public void setAuthorId(Long authorId) { this.authorId = authorId; }}
// AFTER// the above class definition could be replaced with the followingpublic record Article(String title, String content, Long authorId) {}
Sealed Classes
For an in-depth article, see Oracle's "Fight ambiguity and improve your code with Java 17's sealed classes".
Sealed classes can restrict inheritance and extensibility thatttt gives developers an additional tool when designing class hierarchies.
// Example of a three-layer inheritance public class Hero {
}
// TankHero interfacepublic class TankHero extends Hero {
}
// AttachHero interfacepublic class AttackHero extends Hero {
}
// SupportHero interfacepublic class SupportHero extends Hero {
}
// TankHero definitionpublic class Alistar extends TankHero {
}
// AttachHero definitionpublic class Ezreal extends AttackHero {
}
// SupportHero definitionpublic class Soraka extends SupportHero {
}
If we want to restrict the inheritance of Hero class as only permit these three interface (TankHero, AttackHero, SupportHero), we can use a sealed class:
public sealed class Hero permits TankHero, AttackHero, SupportHero {}
The child class of the sealed class can either choose to define sealed
(only permit inheritance to restricted class), non-sealed
(can be inherited without restriction), final
(cannot be inherited).
// if we allow to increase different heros from tankHero interface, // we can use non-sealedpublic non-sealed class TankHero extends Hero {}
Pattern Matching for instanceof
JEP 394: Pattern Matching for instanceof
For an in-depth look, see Oracle's documentation on Pattern Matching for instanceof.
By adding in support for pattern matching on instanceof
statements, Java 17 addresses the "instanceof
-and-cast idiom", by extracting data that has been tested against a particular structure.
See this "instanceof
-and-cast" example:
if (object instanceOf Animal) { Animal animal = (Animal) object; // Cast animal.makeSound();}
We can use the new pattern matching to streamline the above into this example below:
if (object instanceof Animal animal) { animal.makeSound();}
Helpful NullPointerExceptions
JEP 358: Helpful NullPointerExceptions
For a detailed example, see entry on Helpful NullPointerExceptions.
For example with the following Java statement:
a.b.c.i = 99;
Before, a case where the a.b
variable was null would see the following exception thrown:
Exception in thread "main" java.lang.NullPointerException at Prog.main(Prog.java:5)
The above message would be the same if either a
or a.b.c
was the null variable, and is thus not as helpful as developers may want it to be.
Now in Java 17 with the Helpful NullPointerExceptions, a case where the a.b
variable was null would see the following exception thrown:
Exception in thread "main" java.lang.NullPointerException: Cannot read field "c" because "a.b" is null at Prog.main(Prog.java:5)With this improvement, Java's `NullPointerException`s will be much more descriptive.
Highlights for JDK 11
String API Addition
Java11 added a few new methods to the String class: isBlank, lines, strip, stripLeading, stripTrailing, and repeat.
Repeat()
@Testpublic void whenRepeatStringTwice_thenGetStringTwice() { String output = "La ".repeat(2) + "Land";
is(output).equals("La La Land");}
Strip()
returns a string with all leading and trailing whitespaces removed
@Testpublic void whenStripString_thenReturnStringWithoutWhitespaces() { is("\n\t hello \u2005".strip()).equals("hello");}
strip() determines whether the character is whitespace or not based on Character.isWhitespace(). In other words, it is aware of Unicode whitespace characters. This is different from trim(), which defines space as any character that is less than or equal to the Unicode space character (U+0020). If we use trim() in the previous example, we will get a different result:
@Testpublic void whenTrimAdvanceString_thenReturnStringWithWhitespaces() { is("\n\t hello \u2005".trim()).equals("hello \u2005");}
isBlank()
returns true if the string is empty or contains only whitespace. Otherwise, it returns false:
@Testpublic void whenBlankString_thenReturnTrue() { assertTrue("\n\t\u2005 ".isBlank());}
lines()
returns a Stream of lines extracted from the string, separated by line terminators (one of the following: “\n”, “\r”, or “\r\n”.):
@Testpublic void whenMultilineString_thenReturnNonEmptyLineCount() { String multilineStr = "This is\n \n a multiline\n string.";
long lineCount = multilineStr.lines() .filter(String::isBlank) .count();
is(lineCount).equals(3L);}
New File Methods
new readString and writeString static methods from the Files class:
Path filePath = Files.writeString(Files.createTempFile(tempDir, "demo", ".txt"), "Sample text");String fileContent = Files.readString(filePath);assertThat(fileContent).isEqualTo("Sample text");
List to an array
List sampleList = Arrays.asList("Java", "Kotlin");String[] sampleArray = sampleList.toArray(String[]::new);assertThat(sampleArray).containsExactly("Java", "Kotlin");
Predicate.not Method
List<String> sampleList = Arrays.asList("Java", "\n \n", "Kotlin", " ");List withoutBlanks = sampleList.stream() .filter(Predicate.not(String::isBlank)) .collect(Collectors.toList());assertThat(withoutBlanks).containsExactly("Java", "Kotlin");
Local-Variable Syntax for Lambda
Support for using the local variable syntax (var keyword) in lambda parameters was added in Java 11.
We can make use of this feature to apply modifiers to our local variables, like defining a type annotation:
List<String> sampleList = Arrays.asList("Java", "Kotlin");String resultString = sampleList.stream() .map((@Nonnull var x) -> x.toUpperCase()) .collect(Collectors.joining(", "));assertThat(resultString).isEqualTo("JAVA, KOTLIN");
HTTP Client
HttpClient httpClient = HttpClient.newBuilder() .version(HttpClient.Version.HTTP_2) .connectTimeout(Duration.ofSeconds(20)) .build();HttpRequest httpRequest = HttpRequest.newBuilder() .GET() .uri(URI.create("http://localhost:" + port)) .build();HttpResponse httpResponse = httpClient.send(httpRequest, HttpResponse.BodyHandlers.ofString());assertThat(httpResponse.body()).isEqualTo("Hello from the server!");