Skip to content

Commit 7d75f68

Browse files
committed
JDK 24: added example of Class File API
1 parent b4f0c30 commit 7d75f68

5 files changed

+99
-8
lines changed

README.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ A project to explore more about the new features from Java 8 through Java 21.
1111

1212
## Resume by Version
1313

14-
* [Java 24](java-24/)
14+
* [Java 24](java-24/) (Mar, 2025)
1515
* Generational Shenandoah (experimental)
1616
* Compact Object Headers (experimental)
1717
* Prepare to Restrict the Use of JNI
@@ -258,6 +258,7 @@ jshell --enable-preview
258258
* [Learn Java](https://dev.java/learn/)
259259
* [Java Playground](https://dev.java/playground/)
260260
* [Inside.java - Sip of Java](https://inside.java/2021/10/21/sip24/)
261+
* [Learn.java](https://learn.java/)
261262
* Download Java versions:
262263
* [The Role of Preview Features in Java and Beyond](https://blogs.oracle.com/javamagazine/the-role-of-previews-in-java-14-java-15-java-16-and-beyond)
263264
* [Place to get early releases from Oracle's JDK](https://jdk.java.net/)

java-24/BasicClass.java

-3
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,5 @@
11
import java.io.Serializable;
22

3-
/**
4-
* BasicClass
5-
*/
63
public class BasicClass implements Serializable {
74
private String attribute;
85
private int timestamp;

java-24/ClassFileApiReadingExample.java

+6-3
Original file line numberDiff line numberDiff line change
@@ -10,14 +10,17 @@
1010
*/
1111
public class ClassFileApiReadingExample {
1212
public static void main(String[] args) throws Exception {
13-
var classBytes = Files.readAllBytes(Paths.get("BasicClass.class"));
13+
var className = "BasicClass.class";
14+
// var className = "HelloWorldFromClassFile.class";
15+
16+
var classBytes = Files.readAllBytes(Paths.get(className));
1417

1518
// ClassModel is an immutable description of a class file
1619
ClassModel cm = ClassFile.of().parse(classBytes);
1720

18-
// ClasModel is lazy, iterating over it parses the entire class
21+
// ClassModel is lazy, iterating over it parses the entire class
1922
for (ClassElement ce : cm) {
20-
// possible values: https://download.java.net/java/early_access/jdk24/docs/api/java.base/java/lang/classfile/ClassElement.html
23+
// possible values: https://docs.oracle.com/en/java/javase/24/docs/api/java.base/java/lang/classfile/ClassElement.html
2124
switch (ce) {
2225
case Superclass cn -> System.out.println("Superclass: " + cn.superclassEntry().name().stringValue());
2326

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
import java.lang.classfile.*;
2+
import java.lang.constant.*;
3+
import java.nio.file.Files;
4+
import java.nio.file.Paths;
5+
import java.nio.file.OpenOption;
6+
7+
/**
8+
* Run: `java --source 24 --enable-preview ClassFileApiWritingExample.java`
9+
*/
10+
public class ClassFileApiWritingExample {
11+
public static void main(String[] args) {
12+
var className = "HelloWorldFromClassFile";
13+
14+
var CD_System = ClassDesc.of("java.lang.System");
15+
var CD_PrintStream = ClassDesc.of("java.io.PrintStream");
16+
// we can use ClassDesc.of to get String class descriptor or use ConstantDescs with fixed class descriptors
17+
// arrayType returns a class descriptors for array of that type
18+
var MTD_void_StringArray = MethodTypeDesc.of(ConstantDescs.CD_void, ConstantDescs.CD_String.arrayType());
19+
var MTD_void_String = MethodTypeDesc.of(ConstantDescs.CD_void, ClassDesc.of("java.lang.String"));
20+
21+
System.out.println("Building class");
22+
byte[] bytes = ClassFile.of().build(
23+
ClassDesc.of(className),
24+
clb -> clb.withFlags(ClassFile.ACC_PUBLIC)
25+
.withMethod(ConstantDescs.INIT_NAME, ConstantDescs.MTD_void, ClassFile.ACC_PUBLIC,
26+
mb -> mb.withCode(
27+
cob -> cob.aload(0)
28+
// invokes Object constructor
29+
.invokespecial(ConstantDescs.CD_Object, ConstantDescs.INIT_NAME, ConstantDescs.MTD_void)
30+
// calls System.out.println
31+
.getstatic(CD_System, "out", CD_PrintStream)
32+
.ldc("Class " + className + " constructor")
33+
.invokevirtual(CD_PrintStream, "println", MTD_void_String)
34+
.return_()
35+
)
36+
)
37+
// another way to build method without using MethodBuilder.withCode
38+
.withMethodBody("main", MTD_void_StringArray, ClassFile.ACC_PUBLIC + ClassFile.ACC_STATIC,
39+
cob -> cob.getstatic(CD_System, "out", CD_PrintStream)
40+
.ldc("Hello World from class built from Class File API")
41+
.invokevirtual(CD_PrintStream, "println", MTD_void_String)
42+
.return_()
43+
)
44+
);
45+
46+
// writes the class to file
47+
// Inspect with: `javap HelloWorldFromClassFile.class`
48+
// Run with: `java HelloWorldFromClassFile`
49+
try {
50+
System.out.println("Writing class");
51+
Files.write(Paths.get(className + ".class"), bytes);
52+
} catch (Exception ex) {
53+
System.err.println("Error during writing: " + ex.getMessage());
54+
}
55+
56+
// loads the written class
57+
try {
58+
System.out.println("Loading and running class");
59+
var clazz = ClassLoader.getSystemClassLoader().loadClass(className);
60+
var instance = clazz.getConstructors()[0].newInstance();
61+
clazz.getMethods()[0].invoke(instance, new Object[] { args });
62+
} catch (ClassNotFoundException ex) {
63+
System.err.println("Class not found");
64+
} catch (Exception ex) {
65+
System.err.println("Error during running: " + ex.getMessage());
66+
}
67+
}
68+
}

java-24/README.md

+23-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ To run each example use: `java --enable-preview --source 24 <FileName.java>`
5050
* **Late Barrier Expansion for G1**
5151
* reduce the overhead of C2 JIT compiler
5252
* reduce the execution time of C2 when using G1
53-
* **Ahead-of-Time Class Loading & Linkin**
53+
* **Ahead-of-Time Class Loading & Linking**
5454
* improve startup time by making the class instantly available, in a loaded and linked state, when JVM starts
5555
* the application is monitored during a training run to store the loaded and linked classes in a cache
5656
* only the classes loaded from the class path, the module path and JDK itself will be cached
@@ -157,8 +157,30 @@ To run each example use: `java --enable-preview --source 24 <FileName.java>`
157157
* **Structured Concurrency**
158158
* re-preview with no change
159159
160+
### API
161+
162+
* `NumberFormat` supports parsing numbers with correct suffix
163+
* when the formats expects a suffix and `isParseIntegerOnly` is `true`
164+
*
165+
```java
166+
NumberFormat fmt = NumberFormat.getPercentInstance(Locale.US);
167+
fmt.setParseIntegerOnly(true);
168+
fmt.parse("500.55%"); // 5
169+
fmt.parse("700.01%%asdasdada0"); // 7
170+
```
171+
* added new string reader `java.io.Reader.of(CharSequence)`
172+
* expected to be more efficient than `StringReader` in some cases (without convert string and synchronization)
173+
* can receive a `String`, `StringBuilder` and `StringBuffer`
174+
* support for Unicode 16.0
175+
* `Socket.connect` closes the socket if the connection cannot be established
176+
160177
## Links
161178
162179
* [JDK 24 - JEP Dashboard](https://bugs.openjdk.org/secure/Dashboard.jspa?selectPageId=22701)
163180
* [JDK 24 JEPs](https://openjdk.org/projects/jdk/24/)
181+
* [JDK 24 Doc - Classfiles](https://docs.oracle.com/en/java/javase/24/docs/)
182+
* [JDK 24 Release Notes](https://jdk.java.net/24/release-notes)
183+
* Blogs:
184+
* [A Basic Introduction to the Classfile API](https://ifesunmola.com/a-basic-introduction-to-the-classfile-api/#creating-the-personrunner-class)
185+
* [Class File API: Not Your Everyday Java API](https://www.unlogged.io/post/class-file-api-not-your-everyday-java-api)
164186

0 commit comments

Comments
 (0)