반응형
첫번째 해결방안은 앞서 배운 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 MyLogger myLogger;
private final ObjectProvider<MyLogger> myLoggerProvider;
@RequestMapping("log-demo")
@ResponseBody
public String logDemo(HttpServletRequest request){
String requestURL = request.getRequestURI().toString();
MyLogger myLogger = myLoggerProvider.getObject();
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 // 의존관계 주입, final이 붙거나 @NotNull 이 붙은 필드의 생성자를 자동 생성해주는 롬복 어노테이션
public class LogDemoService {
// private final MyLogger myLogger;
private final ObjectProvider<MyLogger> myLoggerProvider;
public void logic(String id) {
MyLogger myLogger = myLoggerProvider.getObject();
myLogger.log("service id = "+ id);
}
}
`main()` 메서드로 스프링을 실행하고, 웹 브라우저에 `http://localhost:8080/log-demo` 를 입력하자.
드디어 잘 작동하는 것을 확인할 수 있다.
[ fa83f638-0f23-4907-af0e-f5cb7b83bd7f ] request scope bean create:hello.core.common.MyLogger@45d4ec60
[ fa83f638-0f23-4907-af0e-f5cb7b83bd7f ][ /log-demo ]controller test
[ fa83f638-0f23-4907-af0e-f5cb7b83bd7f ][ /log-demo ]service id = testID
[ fa83f638-0f23-4907-af0e-f5cb7b83bd7f ] request scope bean close:hello.core.common.MyLogger@45d4ec60
- ObjectProvider` 덕분에 `ObjectProvider.getObject()` 를 호출하는 시점까지 request scope 빈생성을 지연할 수 있다.
(정확한 표현은 스프링 컨테이너한테 요청하는 시간을 지연시킬수 있었다. 그럼 스프링 빈이 나중에 생성을 해주는것) - `ObjectProvider.getObject()` 를 호출하시는 시점에는 HTTP 요청이 진행중이므로 request scope 빈의 생성이 정상 처리된다.
- `ObjectProvider.getObject()` 를 `LogDemoController` , `LogDemoService` 에서 각각 한번씩 따로
호출해도 같은 HTTP 요청이면 같은 스프링 빈이 반환된다! → 내가 직접 이걸 구분하려면 얼마나 힘들까 ㅠㅠ…
이 정도에서 끝내도 될 것 같지만… 개발자들의 코드 몇자를 더 줄이려는 욕심은 끝이 없다.
코드를 더 줄일수 있는 방법이있다 = 프록시
반응형
'Spring > 스프링 핵심 원리 - 기본편' 카테고리의 다른 글
58. 스코프와 프록시 (0) | 2023.08.13 |
---|---|
56. request 스코프 예제 만들기 (0) | 2023.08.13 |
55. 웹 스코프 (0) | 2023.08.13 |
54. 프로토타입 스코프 - 싱글톤 빈과 함께 사용시 Probider로 문제해결 (0) | 2023.08.13 |
53. 프로토타입 스코프 - 싱글톤 빈과 함께 사용시 문제점 (0) | 2023.08.05 |