fetchUser(id)); var orderTask = scope.fork(() -> fetchOrder(id)); scope.join().throwIfFailed(); // if either fails, closes both String result2 = userTask.get().name() + " / " + orderTask.get().total(); } // scope.close() cancels any still-running tasks // Choose CompletableFuture when: // - Integrating with existing CF-based APIs // - You need complex transformation pipelines on the result // - You cannot use preview features // Choose StructuredTaskScope when: // - You want automatic cancellation and no thread leaks // - Code runs on virtual threads (Java 21) // - You want structured, readable concurrent code"> fetchUser(id)); var orderTask = scope.fork(() -> fetchOrder(id)); scope.join().throwIfFailed(); // if either fails, closes both String result2 = userTask.get().name() + " / " + orderTask.get().total(); } // scope.close() cancels any still-running tasks // Choose CompletableFuture when: // - Integrating with existing CF-based APIs // - You need complex transformation pipelines on the result // - You cannot use preview features // Choose StructuredTaskScope when: // - You want automatic cancellation and no thread leaks // - Code runs on virtual threads (Java 21) // - You want structured, readable concurrent code" />

Prev Next

Java / Java 21 Interview Questions

When should you use StructuredTaskScope instead of CompletableFuture in Java 21?

Both APIs manage concurrent asynchronous work, but they have different designs, guarantees, and ideal use cases. Java 21 introduces StructuredTaskScope as the preferred model when running on virtual threads.

CompletableFuture vs StructuredTaskScope
AspectCompletableFutureStructuredTaskScope
API styleCallback/continuation chainImperative — fork, then join
CancellationManual — no automatic cleanupAutomatic — scope closes all tasks on exit
Error propagationManual — must handle in chainthrowIfFailed() re-throws cleanly
Thread modelForkJoinPool (platform threads) or custom executorVirtual threads (one per fork)
StructureUnstructured — tasks can outlive scopeStructured — subtasks never outlive scope
Stack tracesFragmented across continuationsFull linear traces per virtual thread
AvailabilityJava 8+Java 21 (preview)
// CompletableFuture — unstructured, manual cancellation
CompletableFuture  userF  = CompletableFuture.supplyAsync(() -> fetchUser(id));
CompletableFuture orderF = CompletableFuture.supplyAsync(() -> fetchOrder(id));
// If fetchUser throws, fetchOrder may continue running — thread leak possible
String result = userF.thenCombine(orderF,
    (u, o) -> u.name() + " / " + o.total()).join();

// StructuredTaskScope — structured, automatic cleanup
try (var scope = new StructuredTaskScope.ShutdownOnFailure()) {
    var userTask  = scope.fork(() -> fetchUser(id));
    var orderTask = scope.fork(() -> fetchOrder(id));
    scope.join().throwIfFailed();  // if either fails, closes both
    String result2 = userTask.get().name() + " / " + orderTask.get().total();
} // scope.close() cancels any still-running tasks

// Choose CompletableFuture when:
// - Integrating with existing CF-based APIs
// - You need complex transformation pipelines on the result
// - You cannot use preview features

// Choose StructuredTaskScope when:
// - You want automatic cancellation and no thread leaks
// - Code runs on virtual threads (Java 21)
// - You want structured, readable concurrent code
What critical problem does StructuredTaskScope solve that CompletableFuture does not guarantee?
Why do virtual thread stack traces in StructuredTaskScope applications read better than CompletableFuture traces?

Invest now in Acorns!!! 🚀 Join Acorns and get your $5 bonus!

Invest now in Acorns!!! 🚀
Join Acorns and get your $5 bonus!

Earn passively and while sleeping

Acorns is a micro-investing app that automatically invests your "spare change" from daily purchases into diversified, expert-built portfolios of ETFs. It is designed for beginners, allowing you to start investing with as little as $5. The service automates saving and investing. Disclosure: I may receive a referral bonus.

Invest now!!! Get Free equity stock (US, UK only)!

Use Robinhood app to invest in stocks. It is safe and secure. Use the Referral link to claim your free stock when you sign up!.

The Robinhood app makes it easy to trade stocks, crypto and more.


Webull! Receive free stock by signing up using the link: Webull signup.

More Related questions...

What is Java 21 and why is it a significant release? What are Virtual Threads in Java 21 and how do they differ from Platform Threads? How does Pattern Matching for switch work in Java 21? What are Record Patterns in Java 21 and how do they enable deconstruction? What are Sequenced Collections in Java 21? What are sealed classes and interfaces in Java and why are they important for pattern matching? What are Java Records and what do they automatically generate? What are Text Blocks in Java and how do you use them? What is Structured Concurrency in Java 21 and what problem does it solve? What are Scoped Values in Java 21 and how do they differ from ThreadLocal? What is Generational ZGC in Java 21 and why does it improve upon the original ZGC? What important String methods were added from Java 11 through Java 21? What is 'var' in Java and what are its limitations? What are switch expressions in Java and how do they differ from switch statements? How does pattern matching for instanceof work in Java 16+? What Stream API improvements were introduced in Java 9 through Java 21? How has Optional been improved and how should it be used correctly? How does CompletableFuture work in Java and how does it relate to virtual threads? What does the 'volatile' keyword guarantee in Java's memory model? What are the immutable collection factory methods introduced in Java 9? What are functional interfaces in Java and how are lambdas related to them? What are the most important Collectors and how do you write custom ones? What are Unnamed Classes and Instance Main Methods in Java 21 (Preview)? What improvements were made to NullPointerException messages in Java 14? How does type erasure affect instanceof checks with generics in Java? What are the differences between 'synchronized' and ReentrantLock in Java? What are the key classes in the java.time package and when do you use each? What garbage collectors are available in Java 21 and how do you choose between them? What is the Java Platform Module System (JPMS) and when should you use it? What are String Templates in Java 21 (Preview) and how do they improve string interpolation? What are the contracts for equals(), hashCode(), and Comparable in Java? What are the best practices for exception handling in Java? Why is immutability important in Java and how do you implement it correctly? What are default and static methods in Java interfaces? How does HashMap work internally in Java? How do virtual threads compare to reactive programming (Project Reactor / RxJava)? What is the Java 11 HttpClient and how do you use it for HTTP requests? What capabilities do Java enums have beyond simple named constants? What are the java.util.concurrent.atomic classes and how do they work? How does Java Reflection work and what are its performance implications? What are Unnamed Patterns and Variables in Java 21 (Preview) and how do they reduce boilerplate? When should you use StructuredTaskScope instead of CompletableFuture in Java 21? What major APIs were removed between Java 17 and Java 21? What are the most important JVM flags for tuning Java 21 application performance? What are the key steps and pitfalls when migrating an application to Java 21? How do Spring 7 and Spring Boot 4 optimize thread usage compared to older blocking models?
Show more question and Answers...

Spring

Comments & Discussions