첫번째 해결 방안은 Provider를 사용하는 것이다.
간단히 ObjectProvider를 사용해보자.
package hello.core.web;
import hello.core.common.MyLogger;
import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import javax.servlet.http.HttpServletRequest;
@Controller
@RequiredArgsConstructor
public class LogDemoController {
private final LogDemoService logDemoService;
private final ObjectProvider<MyLogger> myLoggerProvider;
@RequestMapping("log-demo")
@ResponseBody
public String logDemo(HttpServletRequest request){
String requestURL = request.getRequestURL().toString();
System.out.println(requestURL);
MyLogger myLogger = myLoggerProvider.getObject(requestURL);
myLogger.setRequestURL(requestURL);
myLogger.log("controller test");
logDemoService.logic("testId");
return "OK";
}
}
package hello.core.web;
import hello.core.common.MyLogger;
import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.stereotype.Service;
@Service
@RequiredArgsConstructor
public class LogDemoService {
private final ObjectProvider<MyLogger> myLoggerProvider;
public void logic(String id) {
MyLogger myLogger = myLoggerProvider.getObject();
myLogger.log("servide id = " + id);
}
}
메인 메소드로 스프링을 실행하고, 웹 브라우저에 localhost:8080/log-demo를 입력하자.
<http://localhost:8080/log-demo>
[3976281e-9c7c-4dc7-9c76-0bca4ac724ee] request scope bean create!hello.core.common.MyLogger@676a181f
[3976281e-9c7c-4dc7-9c76-0bca4ac724ee] [<http://localhost:8080/log-demo>] {controller test}
[3976281e-9c7c-4dc7-9c76-0bca4ac724ee] [<http://localhost:8080/log-demo>] {servide id = testId}
[3976281e-9c7c-4dc7-9c76-0bca4ac724ee] request scope bean close!hello.core.common.MyLogger@676a181f
UUID : [3976281e-9c7c-4dc7-9c76-0bca4ac724ee]
requestURL : [http://localhost:8080/log-demo]
ObjectProvider 덕분에 ObjectProvider.getObject()를 호출하는 시점까지 request scope 빈의 생성을 지연할 수 있다.
ObjectProvider.getObject()를 호출하는 시점에는 HTTP 요청이 진행 중이므로 request scope빈의 생성이 정상적으로 처리된다.
Object.Provider.getObject()를 LogDemoController, LogDemoService에서 각각 한번씩 따로 호출해도 같은 HTTP 요청이면 같은 스프링 빈이 반환된다.