You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
This is done because the latter has a very small
startup overhead (which one can see in a
CPU time flamegraph obtained from a startup
of a REST application)
The proposed change in the code uses StringBuilder instead of the default indyfied string concatenation (introduced in JEP 280). Here's an explanation of why this change is better in this specific context:
Avoids Startup Overhead
The comment in the code mentions avoiding the "small startup cost incurred by JEP 280." The indyfied concatenation relies on invokedynamic (indy) instructions, which introduce a minimal overhead during startup as the JVM needs to bootstrap the necessary method handles. While this cost is negligible in many cases, avoiding it can be beneficial in performance-sensitive scenarios, especially for applications optimized for fast startup, such as Quarkus.
Predictable Performance
StringBuilder is a well-established class in Java, known for its consistent and predictable performance when handling string concatenation. It performs concatenation in-place using a mutable character array, avoiding intermediate object creation.
In contrast, indyfied concatenation dynamically resolves the optimal concatenation strategy at runtime, which may involve creating additional objects or invoking runtime-compilation-like behavior. While efficient in many use cases, it might not always align with the deterministic behavior required for low-level, high-performance code.
Reduced Memory Allocation
By precomputing the required buffer size with new StringBuilder(length):
The code ensures that no unnecessary array resizing or copying occurs during concatenation.
It reduces temporary object creation compared to the potential internal workings of indyfied concatenation.
4. Better Flamegraph Clarity
The comment also points out that the startup CPU flamegraph reveals the cost of indyfied concatenation. By switching to StringBuilder, developers can reduce this noise in profiling tools, making it easier to identify and address other performance bottlenecks.
Trade-Offs
Code Readability: indyfied concatenation (+) is more readable and concise. However, in low-level code paths or performance-critical areas, the verbosity of StringBuilder is a worthwhile trade-off.
Performance Gains Depend on Context: The performance difference might not be noticeable in all cases but is significant in frameworks like Quarkus that prioritize low-latency startup.
Conclusion
Using StringBuilder in this context ensures predictable performance, eliminates startup overhead, and aligns with the performance goals of Quarkus. This change reflects a deliberate optimization for a specific requirement—fast startup and efficient runtime behavior.
The text was updated successfully, but these errors were encountered:
// we want to avoid the small startup cost incurred by JEP 280 and that shows up in the startup cpu flamegraphthis.mediaTypeString = newStringBuilder(mediaType.getType().length() + 1 + mediaType.getSubtype().length())
.append(mediaType.getType()).append("/").append(mediaType.getSubtype()).toString();
this.mediaTypeSubstring = newStringBuilder(mediaType.getType().length() + 2).append(mediaType.getType()).append("/*")
.toString();
Preconditions
As it does hurt readability quite a bit we'd want to be selective where any such recipe gets applied; I'm not sure how to best determine when we'd apply this recipe or not.
Additional context
The stated goal of JEP 280 is outlined as follows:
Change the static String-concatenation bytecode sequence generated by javac to use invokedynamic calls to JDK library functions. This will enable future optimizations of String concatenation without requiring further changes to the bytecode emitted by javac.
Any changes here would then of course mean those future optimizations wont apply if we go for new String Builder instead.
Given the above I wonder if this is even something to write a recipe for, as this type of change is likely only desired where a definite performance issue has been detected, and not something to replicate anywhere we see string concatenation. What are your thoughts here @punkratz312 ?
This is done because the latter has a very small
startup overhead (which one can see in a
CPU time flamegraph obtained from a startup
of a REST application)
See:
The proposed change in the code uses StringBuilder instead of the default indyfied string concatenation (introduced in JEP 280). Here's an explanation of why this change is better in this specific context:
Avoids Startup Overhead
The comment in the code mentions avoiding the "small startup cost incurred by JEP 280." The indyfied concatenation relies on invokedynamic (indy) instructions, which introduce a minimal overhead during startup as the JVM needs to bootstrap the necessary method handles. While this cost is negligible in many cases, avoiding it can be beneficial in performance-sensitive scenarios, especially for applications optimized for fast startup, such as Quarkus.
Predictable Performance
StringBuilder is a well-established class in Java, known for its consistent and predictable performance when handling string concatenation. It performs concatenation in-place using a mutable character array, avoiding intermediate object creation.
In contrast, indyfied concatenation dynamically resolves the optimal concatenation strategy at runtime, which may involve creating additional objects or invoking runtime-compilation-like behavior. While efficient in many use cases, it might not always align with the deterministic behavior required for low-level, high-performance code.
Reduced Memory Allocation
By precomputing the required buffer size with new StringBuilder(length):
The code ensures that no unnecessary array resizing or copying occurs during concatenation.
It reduces temporary object creation compared to the potential internal workings of indyfied concatenation.
4. Better Flamegraph Clarity
The comment also points out that the startup CPU flamegraph reveals the cost of indyfied concatenation. By switching to StringBuilder, developers can reduce this noise in profiling tools, making it easier to identify and address other performance bottlenecks.
Trade-Offs
Code Readability: indyfied concatenation (+) is more readable and concise. However, in low-level code paths or performance-critical areas, the verbosity of StringBuilder is a worthwhile trade-off.
Performance Gains Depend on Context: The performance difference might not be noticeable in all cases but is significant in frameworks like Quarkus that prioritize low-latency startup.
Conclusion
Using StringBuilder in this context ensures predictable performance, eliminates startup overhead, and aligns with the performance goals of Quarkus. This change reflects a deliberate optimization for a specific requirement—fast startup and efficient runtime behavior.
The text was updated successfully, but these errors were encountered: