컴포넌트 스캔과 자동 의존관계 설정
- controller가 service를 의존한다 -> 서로 의존관계가 있다.
//MemberController.java
package hello.hellospring.controller;
import org.springframework.stereotype.Controller;
@Controller
public class MemberController {
private final MemberService memberService = new MemberService();
}
- @Controller라고 명시해주면 spring container에 MemberController 객체를 생성하여 관리 -> 스프링 빈이 관리된다.
- 스프링이 관리를 하게되면 spring container에 등록을 하고 받아쓰는 방식으로 변환해야함
- new를 사용하면 객체가 여러개 생성됨. 이보다는 하나만 생성해서 공유하는 것이 효율적
//MemberController.java
package hello.hellospring.controller;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
@Controller
public class MemberController {
private final MemberService memberService;
@Autowired
public MemberController(MemberService memberService){
this.memberService = memberService;
}
}
- @Autowired: 생성자가 호출될 때 spring container에 있는 memberService와 연결해줌.
- 이 상태에서는 memberService를 spring이 알 수 있는 방법이 없음.
//MemberService.java
package hello.hellospring.service;
import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import org.springframework.stereotype.Service;
import java.util.List;
import java.util.Optional;
@Service /*추가*/
public class MemberService {
. . .
}
- @Service : 서비스임을 인지하고 spring container에 서비스를 등록
//MemoryMemberRepository.java
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import org.springframework.stereotype.Repository;
import java.util.*;
@Repository /*추가*/
public class MemoryMemberRepository implements MemberRepository{
. . .
}
- @Repository:리포지토리임을 인지하고 spring container에 리포지토리를 등록
스프링 빈을 등록하는 두가지 방법
- 컴포넌트 스캔과 자동 의존관계 설정
- 컴포넌트 스캔 : @Service, @Repository,@Controller 모두 자세히 보면 @Component annotation을 포함하고 있음.
- @Component annotation이 있다면 spring이 객체를 생성해 spring container에 저장(스프링 빈으로 자동 등록)
- 실행시키는 파일을 포함한 패키지와 동일선상 또는 하위 패키지가 아니라면 스프링빈으로 컴포넌트 스캔을 하지 않음.
- 컴포넌트 스캔 : @Service, @Repository,@Controller 모두 자세히 보면 @Component annotation을 포함하고 있음.
- 자바 코드로 직접 스프링 빈 등록하기
참고: 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤으로 등록한다(유일하게 하나만 등록해서 공유한다) 따라서 같은 스프링 빈이면 모두 같은 인스턴스다. 설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용한다.
자바 코드로 직접 스프링 빈 등록하기
//SpringConfig.java
package hello.hellospring;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import hello.hellospring.service.MemberService;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class SpringConfig {
@Bean
public MemberService memberService() {
return new MemberService(memberRepository());
}
@Bean
public MemberRepository memberRepository() {
return new MemoryMemberRepository();
}
}
- @Configuration : 스프링이 스프링빈에 등록하라는 것으로 인식
- @Controller, @Autowired는 지우지말고 그대로 두기
DI의 3가지 방법
- 필드 주입
@Controller
public class MemberController {
@Autowired private MemberService memberService;
}
- 추천하지 않는 방법
- 중간에 바꿀 수 있는 방법이 없어서
2. setter 주입
@Controller
public class MemberController {
private MemberService memberService;
@Autowired
public void setMemberService(MemberService memberService){
this.memberService = memberService;
}
}
- 생성은 생성대로 한 후, setter를 통해 나중에 주입
- setMemberService는 바뀔 필요가 없는 함수인데 public으로 선언되어 변경될 위험이 있음.
3. 생성자 주입
- 생성자를 통한 주입(위에 예제)
- 가장 추천하는 방법
정형화되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링빈으로 등록한다.
ex) MemoryMemberRepository를 변경해야할 일이 있을 때 다른 코드를 손댈 필요 없이 config의 코드만 수정해주면 됨
주의: @Autowired 를 통한 DI는 helloController , memberService 등과 같이 스프링이 관리하는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.
ex) MemberService memberService = new MemberService();
'개발 스터디 > Spring' 카테고리의 다른 글
[스프링 핵심원리 - 기본편] 객체 지향 설계와 스프링 (0) | 2023.05.02 |
---|---|
[Spring 스터디]웹 MVC 개발 (0) | 2023.04.08 |
[Spring 스터디] 스프링 예제 (회원 관리 서비스) (0) | 2023.04.05 |
[Spring 스터디] 웹 개발 기초 (0) | 2023.04.04 |
[Spring 스터디] 스프링 프로젝트 생성하기 (feat. 프로젝트 생성 오류 해결) (0) | 2023.04.04 |