기본 콘텐츠로 건너뛰기

[Spring] DI 1탄 - DI개요

DI(Dependency Injection)란 무엇인가?

기능을 실행하기 위해 다른 클래스를 필요로 할 때 이를 의존한다 한다.


public class UserServiceImpl implements UserService {
      private final UserRepository userRepository;
      private final PasswordEncoder passwordEncoder;

      public UserServiceImpl(DataSource dataSource) {
          this.userRepository = new JdbcUserRepository(dataSource);
          this.passwordEncoder = new BCryptPasswordEncoder();
      }
      @Override
      public void register(User user, String rawPassword) {

      }
}


생성자에서 위와 같이 초기화를 직접 할당하게 되면 '클래스간 결합도가 높다'라는 문제가
발생한다. 이는 UserServiceImple을 개발하는 단계에서 의존하는 컴포넌트 클래스가
완성되어 있어야 한다는 의미다.


의존하는 클래스가 아직 개발중이거나 미완성이라면 완벽한 컴포넌트를 구성할 수가 없게 된다.


이런 결합도를 낮추려면 생성자에서 직접생성 대신 생성자의 인수로 받아 할당하는 방법이 있다.
 
public UserServiceImpl(UserRepository userRepository, PasswordEncoder passwordEncoder) {
      this.userRepository = userRepository;
      this.passwordEncoder = passwordEncoder;
}


어떤 클래스가 필요로 하는 컴포넌트를 외부에서 생성한 후, 내부에서 사용 가능하게 만들어주는
과정을 '의존성 주입(DI)'이라 한다.
이런 의존성 주입을 자동으로 처리하는 기반을
'DI컨테이너(=스프링컨테이너 = IoC컨테이너 = 빈컨테이너)'라고 한다.
인스턴스의 스코프 관리도 'DI 컨테이너'가 대신해준다.


DI는 IoC라는 디자인 패턴 중 하나다. 컴포넌트를 구성하는 인스턴스의 생성과 의존 관계의
연결 처리를 해당 소스코드가 아닌 DI컨테이너(IoC 컨테이너)에서 대신하기 때문에 제어가
역전 됬다고 한다.


DI에 의존 객체를 전달하는 방법

크게 생성자 방식과 프로퍼티 설정 방식이 있다. 생성자 방식은 앞에서 예제에 봤고, 프로퍼티 방식을
보면, 객체를 전달 받기 위해 매서드(세터메서드)를 이용한다. setXXX.


DI 컨테이너란 무엇인가?

스프링은 객체를 생성하고 객체를 연결해주는 조립기 역할을 한다.
조립기 기능을 하는 클래스중 하나의 예로 GenericXmlApplicationContex가 있다.
조립기 클래스는 xml등의 정의된 설정 정보를 읽어와 객체를 생성, 연결한 뒤 내부적으로 보관한다.
이런 생성과 보관 떄문에 스프링을 ‘객체 컨테이너’라고도 부른다.
스프링 컨테이너가 생성해 보관하는 객체를 스프링 빈 객체라 부른다.

지금까지 DI에 관해서 간략하게 알아보았다.
프로그램이건 어떤 학문을 배울 땐 용어가 가장 중요한거 같다. 앞으로 중요한 용어에 대해선 노란색으로 색칠해 놓을 테니 이해가 부족하다면 다른 참고자료들을 활용해 꼭 숙지하도록 하자.

댓글

이 블로그의 인기 게시물

[자바 웹 프로그래밍]2장 문자열 계산기 구현을 통한 테스트와 리펙토링

이번엔 2장에 나와 있는 내용 정리와 느낀점을 정리 해 보겠다. 1. main() 메소드를 활용한 테스트의 문제점.   - 소스코드 구현 후 정상적으로 동작하는지 확인 위해 일반적인 방법은 main()메소드를 활용하는 것이다.   - 실제 서비스를 담당하는 프로덕션 코드와 이 프로덕션 코드가 정상 동작 하는지 확인을 위한 main() 으로 나뉜다.   - 이 방법의 첫번째 문제점은 프로덕션코드와 main() 메서드가 함께 있다는 것이다.   - 프로덕션 코드와 테스트코드(main)을 분리할 수 있다.   - 두 번째 문제는 내가 구현하고 있는 메서드만 집중 할 수 없고, 클래스가 가지고 있는 모든 메서드를 테스트 할 수 밖에 없다.   - 다른 문제는 항상 콘솔로 확인을 할 수 밖에 없다는 것이다.   - 이를 위해 등장한 라이브러리가 JUnit 이다. 내 관심을 가지는 메서드에 대해 테스트 가능하다. 2. JUnit을 활용해 main() 메서드 문제 극복 2.1 한 번에 메서드 하나에만 집중.   - JUnit관련 라이브러리 추가 후  테스트 메서드에 @Test를 붙이면 된다.   - test 관련 코드 작성 후 Run > Run as> JunitTest를 실행해 보자.   - 각각 테스트 메서드를 독립적으로 실행할 수 있기 때문에 현재 내가 구현하고 있는 프로덕션 코드의 메서드에 집중할 수 있다. import org.junit.Test; public class CalculatorTest { @Test public void add() { Calculator cal = new Calculator(); System.out.println(cal.add(1,2)); } } 2.2 결과 값을 눈이 아닌 프로그램을 통해 자동화 import org.junit.Test; import static org.junit.Assert.assertEquals;

[고량주] 라오왕 연태고량주 플러스

나에게 처음 고량주란 이런것이다 라는걸 알려준 녀석이다. 부모님이 중국집을 하다 보니 가끔 초록색병 고량주를 먹었을때  역한 공업용 알콜 맛에 고량주는 나랑 안맞는다 생각했다가 우연히 양고기에 이녀석을 접한 뒤로 고량주의 맛을 알아버렸다... 제품명 : 라오왕 연태고량주플러스 제품유형 : 일반증류주 도수 : 34.2% 가격 : 9000원(홈플러스 익스프레스 기준) 재구매 의사 : 있다 시음평 : 역시 고량주 특유의 향인데, 열대과일 향도나고, 배향, 살짝 달달한 향이 난다.            목넘김은 34.2%에도 불구하고 그리 힘들지 않았다(주당이 된걸수도..)             중국요리나 양꼬치집에서 맛있는 술이 땡긴다면 강력추천한다.