I'm trying to make a small REST using Spring Boot. I've never used Spring and used Java a long time ago (Java 7)!
In the last 2 years I have used only Python and C# (but like I said, I already used Java).
So, now, I'm trying to make a REST using async methods, and checked several examples, but still, I don't understand very well the "correct way" to do this.
Looking at the following documentation: http://carlmartensen.com/completablefuture-deferredresult-async, Java 8 has CompletableFuture
that I can use with Spring, so, I made the following code:
Service:
@Service public class UserService { private UserRepository userRepository; // dependency injection // don't need Autowire here // https://docs.spring.io/spring-boot/docs/current/reference/html/using-boot-spring-beans-and-dependency-injection.html public UserService(UserRepository userRepository) { this.userRepository = userRepository; } @Async public CompletableFuture<User> findByEmail(String email) throws InterrupedException { User user = userRepository.findByEmail(email); return CompletableFuture.completedFuture(user); } }
Repository:
public interface UserRepository extends MongoRepository<User, String> { @Async findByEmail(String email); }
RestController:
@RestController public class TestController { private UserService userService; public TestController(UserService userService) { this.userService = userService; } @RequestMapping(value = "test") public @ResponseBody CompletableFuture<User> test(@RequestParam(value = "email", required = true) String email) throws InterruptedException { return userService.findByEmail(email).thenApplyAsync(user -> { return user; }) } }
This code give me the expected output. Then, looking at another documentation (sorry, I lost the link), I see that Spring accept the following code (which give me the expected output too):
@RequestMapping(value = "test") public @ResponseBody CompletableFuture<User> test(@RequestParam(value = "email", required = true) String email) throws InterruptedException { return userService.findByEmail(email); } }
Is there a difference between the two methods?
Then, looking at the following guide: https://spring.io/guides/gs/async-method/, there's a @EnableAsync
annotation in SpringBootApplication
class. If I include the @EnableAsync
annotation and create a asyncExecutor
Bean like the code from last link, my application don't return nothing on /test
endpoint (only a 200 OK response, but with blank body).
So, my rest is async without the @EnableAsync
annotation? And why when I use @EnableAsync
, the response body is blank?
The response body is blank because the @Async
annotation is used at findEmail method of UserRepository class, it means that there is no data returned to the following sentence User user = userRepository.findByEmail(email);
because findByEmail method is running on other different thread and will return null instead of a List object.
The @Async
annotation is enabled when you declare @EnableAsync
that is the reason why it only happens when you use @EnableAsync
because it activates the @Async of findEmail method to run it on other thread.
The method return userService.findByEmail(email);
will return a CompletableFuture
object that is created from UserService
class.
The difference with the second method call is that thenApplyAsync
method will create a totally new CompletableFuture
from the previous one that comes from userService.findByEmail(email)
and will only return the user object that comes from the first CompletableFuture
.
return userService.findByEmail(email).thenApplyAsync(user -> { return user; })
If you want to get the expected results just remove the @Async
annotation from findByEmail method, and finally add the @EnableAsync
Annotation
If you need to clarify ideas of how to use Async methods, lets say that you have to call three methods and each one takes 2 seconds to finish, in a normal scenario you will call them method1, then method2 and finally method3 in that case you entire request will take 6 seconds. When you activate the Async approach then you can call three of them and just wait for 2 seconds instead of 6.
Add this long method to user service:
@Async public CompletableFuture<Boolean> veryLongMethod() { try { Thread.sleep(2000L); } catch (InterruptedException e) { e.printStackTrace(); } return CompletableFuture.completedFuture(true); }
And call it three times from Controller, like this
@RequestMapping(value = "test") public @ResponseBody CompletableFuture<User> test(@RequestParam(value = "email", required = true) String email) throws InterruptedException { CompletableFuture<Boolean> boolean1= siteService.veryLongMethod(); CompletableFuture<Boolean> boolean2= siteService.veryLongMethod(); CompletableFuture<Boolean> boolean3= siteService.veryLongMethod(); CompletableFuture.allOf(boolean1,boolean2,boolean3).join(); return userService.findByEmail(email); }
Finally measure the time that takes your response, if it takes more than 6 seconds then you are not running Async method, if it takes only 2 seconds then you succeed.
Also see the following documentation: @Async Annotation, Spring async methods, CompletableFuture class
Hope it help.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With