Spring Component에서 static 변수 및 메소드 사용하는 방법
ApplicationContext를 Util 클래스에서 가지고 와서 사용을 하고 싶었다. 그래서 다음과 같이 ApplicationContextAware를 impl하여 ApplicationContextProvider를 만들었다.
@Component
public class ApplicationContextProvider implements ApplicationContextAware {
private static ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext ctx) throws BeansException {
applicationContext = ctx;
}
public static ApplicationContext getApplicationContext() {
return applicationContext;
}
}
ApplicationContextProvider.getApplicationContext()을 하면 어디서든 ApplicationContext를 가지고 올 수 있도록 만들었고 문제없이 잘 동작하였다.
하지만! Sonarqube를 돌려보니 이건 위험한 코드니 고쳐야 한다고 한다.
Correctly updating a static field from a non-static method is tricky to get right and could easily lead to bugs if there are multiple class instances and/or multiple threads in play. Ideally, static fields are only updated from synchronized static methods.
This rule raises an issue each time a static field is updated from a non-static method.
조금 찾아보니 위의 코드를 멋지게 리팩토링하신 분이 있었다.
@Component
public class ApplicationContextProvider implements ApplicationContextAware {
private static class AplicationContextHolder {
private static final InnerContextResource CONTEXT_PROV = new InnerContextResource();
private AplicationContextHolder() {
super();
}
}
private static final class InnerContextResource {
private ApplicationContext context;
private InnerContextResource() {
super();
}
private void setContext(ApplicationContext context) {
this.context = context;
}
}
public static ApplicationContext getApplicationContext() {
return AplicationContextHolder.CONTEXT_PROV.context;
}
@Override
public void setApplicationContext(ApplicationContext ac) {
AplicationContextHolder.CONTEXT_PROV.setContext(ac);
}
}
ApplicationContextHolder라는 것을 만들고 그 안에 InnerContextResource에서 ApplicationContext를 set한다. 이렇게 캡슐화를 하는것은 어떻게 해야지 실력이 느는걸까... 대단하다.
끝!