-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathJavaBasedConfiguration.kt
111 lines (89 loc) · 6.99 KB
/
JavaBasedConfiguration.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
package ioc
import io.kotest.core.spec.style.FreeSpec
import io.kotest.matchers.shouldBe
import io.kotest.matchers.shouldNotBe
import javaconfig.FullModeConfig
import javaconfig.JavaConfig
import javaconfig.LiteModeService
import org.springframework.context.annotation.AnnotationConfigApplicationContext
import org.springframework.web.servlet.ViewResolver
/**
* Java 기반 컨테이너 구성(Java-based Container Configuration)
*
* 스프링의 Java 기반 컨테이너 구성은 @Configuration 클래스와 @Bean 메서드로 이루어집니다.
* @Bean 메서드는 스프링 IoC 컨테이너에서 관리할 새 객체를 인스턴스화하여 구성 및 초기화에 사용됩니다.
* @Bean은 XML 구성의 <bean/> 요소와 동일한 역할을 합니다.
* 일반적으로 @Configuration 클래스와 함께 사용됩니다.
*
* @Configuration를 클래스에 붙이면 이 클래스의 주요 목적은 Bean 정의를 뜻하게 됩니다.
* 또한 @Configuration 클래스를 사용하면 동일한 클래스에서 다른 @Bean 메서드를 호출하여 Bean 의존성을 정의할 수 있습니다.
*
* 앞선 학습 테스트에서 Java 기반 컨테이너 구성을 사용하여 스프링 IoC 컨테이너를 구성하는 방법을 살펴보았습니다.
* 여기서는 Java 기반 컨테이너 구성의 두 가지 모드인 전체(Full) 모드와 라이트(Lite) 모드에 대해 알아보겠습니다.
* 특별한 이유가 없다면 전체 모드를 사용하는 것이 좋으며 라이트 모드를 사용한다면 어떤 점을 주의해야 하는지 알아보겠습니다.
*/
class JavaBasedConfiguration : FreeSpec({
"""
전체(Full) @Confriguration Mode와 라이트(Lite) @Bean Mode
@Configuration이 붙지 않은 클래스에서 @Bean 메서드를 선언하면 "라이트(Lite)" 모드로 동작합니다.
라이트 모드에서 @Bean 메서드는 해당 클래스의 인스턴스를 생성하는 데 사용됩니다.
주요 목적은 일반적인 Bean과 다르며, @Bean 메서드로 추가적인 기능을 제공하는 용도로 사용합니다.
예를 들어, 특정 서비스 클래스가 다양한 형태의 ViewResolver를 제공해야 하는 경우를 생각해봅시다.
이 때, 서비스 클래스는 @Bean 메서드를 사용하여 ViewResolver 인스턴스를 생성하고 이를 스프링 컨테이너에 제공할 수 있습니다.
라이트 모드에서 선언한 @Bean 메서드는 Bean 간의 의존성을 선언할 수 없습니다.
라이트 모드의 @Bean 메서드는 다른 @Bean 메서드를 호출하면 안 됩니다.
라이트 모드의 장점은 런타임에 CGLIB 서브클래싱을 적용할 필요가 없어 클래스 설계에 제약이 없습니다.
특별한 이유가 없다면 @Configuration 클래스 내에서 @Bean 메서드를 선언하여 항상 "전체(Full) 모드"로 사용합시다.
""" - {
"""
@Service 클래스에서 선언한 @Bean 메서드는 라이트 모드로 동작합니다.
라이트 모드지만 싱글톤 Bean으로 등록되어서 스프링 컨테이너가 동일한 객체를 반환합니다.
""" {
val applicationContext = AnnotationConfigApplicationContext(JavaConfig::class.java)
val firstLiteBean = applicationContext.getBean("viewResolver", ViewResolver::class.java)
val secondLiteBean = applicationContext.getBean("viewResolver", ViewResolver::class.java)
firstLiteBean shouldNotBe secondLiteBean
}
"""
@Serivce 클래스에서 선언한 @Bean 메서드는 라이트 모드로 동작합니다.
LiteModeService 클래스에서 @Bean을 사용한 anotherViewResolver() 메서드는 viewResolver() 메서드를 호출하고 있습니다.
하지만 라이트 모드에서는 Bean 간의 의존성을 선언할 수 없습니다.
anotherViewResolver() 메서드 내의 viewResolver() 메서드는 새로운 ViewResolver 인스턴스를 생성합니다.
싱글톤 스코프이지만 동일한 Bean을 반환하지 않습니다.
""" {
val applicationContext = AnnotationConfigApplicationContext(JavaConfig::class.java)
val viewResolver = applicationContext.getBean("viewResolver", ViewResolver::class.java)
// anotherViewResolver() 메서드 내부에서 viewResolver() 메서드를 호출하더라도 동일한 Bean을 반환하지 않습니다.
val anotherViewResolver = applicationContext.getBean("anotherViewResolver", ViewResolver::class.java)
viewResolver shouldBe anotherViewResolver
}
"""
@Configuration 클래스에서 선언한 @Bean 메서드는 전체 모드로 동작합니다.
FullModeConfig 클래스의 anotherViewResolver() 메서드는 LiteModeService 클래스의 anotherViewResolver() 메서드와 동일한 로직입니다.
하지만 FullModeConfig 클래스의 anotherViewResolver() 메서드는 싱글톤 Bean으로 등록됩니다.
""" {
val applicationContext = AnnotationConfigApplicationContext(FullModeConfig::class.java)
val viewResolver = applicationContext.getBean("viewResolver", ViewResolver::class.java)
// anotherViewResolver() 메서드 내부에서 viewResolver() 메서드를 호출하면 동일한 Bean을 반환합니다.
val firstAnotherViewResolver = applicationContext.getBean("anotherViewResolver", ViewResolver::class.java)
// FullModeConfig 클래스의 anotherViewResolver() 메서드는 싱글톤 Bean으로 등록됩니다.
viewResolver shouldNotBe firstAnotherViewResolver
val fullModeConfig = applicationContext.getBean("fullModeConfig", FullModeConfig::class.java)
// @Configuration 클래스인 fullModeConfig Bean으로 직접 anotherViewResolver() 메서드를 호출해도 동일한 Bean을 반환합니다.
val secondAnotherViewResolver = fullModeConfig.anotherViewResolver()
firstAnotherViewResolver shouldNotBe secondAnotherViewResolver
}
"""
@Configuration 클래스와 달리 @Service 클래스에서 선언한 @Bean 메서드는 단순 팩터리 메서드로 동작합니다.
라이트 모드에서 선언한 @Bean 메서드는 Bean 간의 의존성을 선언할 수 없습니다.
@Configuration 클래스 내에서 선언한 @Bean 메서드는 클래스 내 다른 메서드에서 싱글톤 Bean으로 참조할 수 있습니다.
""" {
val applicationContext = AnnotationConfigApplicationContext(JavaConfig::class.java)
val liteBean = applicationContext.getBean("viewResolver", ViewResolver::class.java)
val service = applicationContext.getBean(LiteModeService::class.java)
val notBean = service.anotherViewResolver()
// @Configuration 클래스에서 선언한 @Bean 메서드와 달리 @Service 클래스에서 선언한 @Bean 메서드는 단순 팩터리 메서드로 동작합니다.
liteBean shouldBe notBean
}
}
})