제어의 역전 (IoC)란 기존의 프로그램에서 구현 객체가 프로그램의 제어흐름을 스스로 조종할 수 있었던 부분을 없애고 구현 객체는 자신의 로직만을 실행하는 역할을 하도록 하는 것이다.
Spring 프로젝트를 보면 흔히 AppConig를 만들어 프로그램의 제어 흐름을 담당하도록 한다. 모든 Implementation 객체들은 필요한 인터페이스를 호출하지만 어떤 구현 객체들이 실행될지는 알지 못하도록 하고 오직 AppConfig가 구현객체를 생성하고 프로그램의 흐름을 직접 제어하는것이 제어의 역전 (IoC)이다.
의존관계에는 정적인 클래스 의존 관계와 실행하는 시점에 결정되는 동적인 객체(인스턴스) 의존 관계이다.
정적인 클래스 의존관계는 클래스가 사용하는 import 코드만 보고 의존관계를 판단할 수 있는 것이다. 동적인 객체 인스턴스 의존관계는 애플리케이션 실행 시점에 실제 생성된 객체 인스턴스의 참조가 연결된 의존 관계이다. 애플리케이션 실행시점에 외부에서 실제 구현 객체를 생성하고 클라이언트에 전달해서 클라이언트와 서버의 실제 의존관계가 연결되는 것을 의존관계 주입이라 한다.
AppConfig처럼 객체를 생성하고 관리하면서 의존관계를 연결해 주는 것을 IoC 컨테이너, DI컨테이너라고 한다.
스프링 컨테이너는 @Configuration이 붙은 AppConfig를 설정 정보로 사용한다. 여기서 @Bean이라 적힌 메소드를 모두 호출해서 반환된 객체를 스프링 컨테이너에 등록한다. 이렇게 컨테이너에 등록된 객체를 스프링 빈이라고 한다.
스프링 빈은 @Bean 이 붙은 메소드의 명을 스프링 빈의 이름으로 사용한다. 이전에는 개발자가 필요한 객체를 AppConfig를 사용해서 직접 조회했지만, 스프링을 사용하면 스프링 컨테이너를 통해서 필요한 스프링빈을 찾아야한다. 스프링 빈은 applicationContext.getBean()메소드를 사용해서 찾을 수 있다.
new AnnotationConfigApplicationContext(AppConfig.class)
를 통해 생성할 수 있다. 스프링 컨테이너를 생성할 때는 구성 정보를 지정해주어야 한다. (ex AppConfig.class)
BeanFactory
스프링 컨테이너의 최상위 인터페이스이다. 스프링 빈을 관리하고 조회하는 역할을 담당한다. getBean()을 제공한다.
ApplicationContext
BeanFactory기능을 모두 상속받아 제공한다. 애플리케이션을 개발할 때는 조회하는 기능은 물론 부가기능이 필요함, 추가적으로 메세지 소스를 활용한 국제화 기능과, 환경변수, 에플리케이션 이벤트, 편리한 리소스 조회 기능 등을 포함한다.
이러한 BeanFactory, ApplicationContext를 스프링 컨테이너라 한다.