Asynchronous approach
With the high development of hardware & software, modern applications become much more complex and demanding. Due to high demand engineers always try to find new ways to improve their application performance and responsiveness. One solution to slow-paced applications is the implementation of the Asynchronous approach. Asynchronous processing is a technique that is a process or function that executes a task to run concurrently, without waiting for one task to complete it before starting another. In this article, I will try to explore the Asynchronous approach and @Async annotation in Spring Boot, trying to explain the differences between multi threading and concurrency, and when to use or avoid it.
Table of contents
Open Table of contents
What is @Async in Spring?
The @Async annotation in Spring enables asynchronous processing of a method call. It instructs the framework to execute the method in a separate thread, allowing the caller to proceed without waiting for the method to complete. This improves the overall responsiveness and throughput of an application.
To use @Async, you must first enable asynchronous processing in your application by adding the @EnableAsync annotation to a configuration class:
@Configuration
@EnableAsync
public class AppConfig {
}
Next, annotate the method you want to execute asynchronously with the @Async annotation:
@Service
public class AsyncService {
@Async
public void asyncMethod() {
// Perform time-consuming task
}
}
How is @Async different from multihreading and Concurrency?
Sometimes It might seem confusing to differentiate multithreading and concurrency from parallel execution, however, both are related to parallel execution. Each of them has their use case and implementation:
-
@Async annotation is Spring Framework specific abstraction, which enables asynchronous execution. It gives the ability to use async with ease, handling all hard work in the background, such as thread creation, management, and execution. This allows users to focus on business logic rather than low-level details.
-
Multithreading is a general concept, commonly referring to the ability of an OS or program to manage multiple threads concurrently. As @Async helps us to do all hard work automatically, in this case, we can handle all this work manually and create a multihreading environment. Java has necessary classes such as Thread and ExecutorService to create and work with multihreading.
-
Concurrency is a much broader concept, and it covers both multihreading and parallel execution techniques. It is the ability of a system to execute multiple tasks simultaneously, on a one or more processors across.
In summary, @Async is a higher-level abstraction that simplifies asynchronous processing for developers, on the other hand multihreading and concurrency is more about to manual management of parallel execution.
When to use @Async and when to avoid it.
It seems very intuitive to use asynchronous approach, however, it must be take into account, there’s do’s and don’ts for this approach as well.
Use @Async when:
- You have independent, time-consuming tasks that can run concurrently without affecting the application’s responsiveness.
- You want a simple and clean way to enable asynchronous processing without diving into low-level thread management.
Avoid using @Async when:
- The tasks you want to execute asynchronously have complex dependencies or need a lot of coordinating. In such cases, you might need to use more advanced concurrency APIs, like CompletableFuture or reactive programming libraries like Project Reactor.
- You must have precise control over how threads are managed., such as custom thread pools or advanced synchronization mechanisms. In these cases, consider using Java’s ExecutorService or other concurrency utilities.
Using @Async in a Spring Boot application.
In this example, we will create a simple Spring Boot application that demonstrates the use of @Async. Let’s create a simple Order management service.
-
Create a new Spring Boot project with minimum dependency requirement:
org.springframework.boot:spring-boot-starter
org.springframework.boot:spring-boot-starter-web
Web dependency is for REST endpoint demonstration purpose. @Async comes with boot starter. -
Add the @EnableAsync annotation to the main class or Application Config class, if we are using it.:
@SpringBootApplication
@EnableAsync
public class AsyncDemoApplication {
public static void main(String[] args) {
SpringApplication.run(AsyncDemoApplication.class, args);
}
}
@Configuration
@EnableAsync
public class ApplicationConfig {}
- For the optimal solution, what we can do is, create a custom Executor bean and customize it as per our needs in the same Configuration class:
@Configuration
@EnableAsync
public class ApplicationConfig {
@Bean
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(5);
executor.setMaxPoolSize(10);
executor.setQueueCapacity(100);
executor.setThreadNamePrefix("AsyncThread-");
executor.initialize();
return executor;
}
}
With this configuration we have control over max and default thread pool size. As well as other useful customizations.
- Create an OrderService class with @Async methods:
@Service
public class OrderService {
@Async
public void saveOrderDetails(Order order) throws InterruptedException {
Thread.sleep(2000);
System.out.println(order.name());
}
@Async
public CompletableFuture<String> saveOrderDetailsFuture(Order order) throws InterruptedException {
System.out.println("Execute method with return type + " + Thread.currentThread().getName());
String result = "Hello From CompletableFuture. Order: ".concat(order.name());
Thread.sleep(5000);
return CompletableFuture.completedFuture(result);
}
@Async
public CompletableFuture<String> compute(Order order) throws InterruptedException {
String result = "Hello From CompletableFuture CHAIN. Order: ".concat(order.name());
Thread.sleep(5000);
return CompletableFuture.completedFuture(result);
}
}
What we did here is create 3 different Async methods. First saveOrderDetails
service is a straightforward asynchronous
service,
which will start doing the computing asynchronously. If we want to use modern asynchronous Java features
like CompletableFuture
, we can achieve it with saveOrderDetailsFuture
service. With this service, we can call a
thread
to wait for the result of
an @Async. It should be noted that CompletableFuture.get()
will block until the result is available. If we want to
perform further asynchronous operations when the result is available, we can use thenApply
, thenAccept
, or other
methods provided by CompletableFuture.
- Create a REST controller to trigger the asynchronous method:
@RestController
public class AsyncController {
private final OrderService orderService;
public OrderController(OrderService orderService) {
this.orderService = orderService;
}
@PostMapping("/process")
public ResponseEntity<Void> process(@RequestBody Order order) throws InterruptedException {
System.out.println("PROCESSING STARTED");
orderService.saveOrderDetails(order);
return ResponseEntity.ok(null);
}
@PostMapping("/process/future")
public ResponseEntity<String> processFuture(@RequestBody Order order) throws InterruptedException, ExecutionException {
System.out.println("PROCESSING STARTED");
CompletableFuture<String> orderDetailsFuture = orderService.saveOrderDetailsFuture(order);
return ResponseEntity.ok(orderDetailsFuture.get());
}
@PostMapping("/process/future/chain")
public ResponseEntity<Void> processFutureChain(@RequestBody Order order) throws InterruptedException, ExecutionException {
System.out.println("PROCESSING STARTED");
CompletableFuture<String> computeResult = orderService.compute(order);
computeResult.thenApply(result -> result).thenAccept(System.out::println);
return ResponseEntity.ok(null);
}
}
Now, when we access the /process
endpoint, the server will return a response right away, while
the saveOrderDetails()
continues to execute in the background. After 2 seconds, the service will complete. Second endpoint - /process/future
will use our second option which is CompletableFuture
In this case after 5 seconds, the service will complete, and
store
the result in CompletableFuture
we can further use future.get()
to access the result.
In the last endpoint - /process/future/chain
, we optimized and used asynchronous computations. Controller using the
same
service method for CompletableFuture
, however right after the future, we are using thenApply
, thenAccept
methods.
The server returns a response right away, we do not need to wait for 5 seconds, and computation will be done background.
The most
important point, in this case, is a call to async service, in our case compute()
must be done from the outside of
the same class. If we use @Async on a method and call it within the same class, it won’t work. This is because Spring
uses
proxies to add asynchronous behavior, and calling the method internally bypasses the proxy. To make it work, we can
either:
- Move the @Async methods to a separate service or component.
- Use ApplicationContext to get the proxy and call the method on it.
Conclusion
The @Async annotation in Spring is a powerful tool for enabling asynchronous processing in application. By using @Async, we don’t need to go into the complexities of concurrency management and multihreading to enhance the responsiveness and performance of our application. But in order to decide when to use @Async or go with alternative concurrency utilities, it’s important to know its limitations and use cases. This is the link for the project used on this blog.