스프링 빈과 의존관계 04 by 김영한

공부한것 다 기록해·2023년 5월 10일
0
post-thumbnail

스프링 빈과 의존관계

스프링 빈을 등록하고, 의존 관계 설정하기

회원 컨트롤러가 회원서비스와 회원 리포지토리를 사용할 수 있게 의존관계를 준비

회원 컨트롤러에 의존 관계 추가

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 { // Spring이 MemberController의 객체를 생성해놓는다.
    private final MemberService memberService;

    @Autowired
    public MemberController(MemberService memberService) {
        this.memberService = memberService;
    }
}
  • @Controller annotation이 있는 경우, 스프링이 딱 뜰때, 생성을 해서 스프링내에서 관리를 해준다.
  • 생성자에 @Autowired가 있는 경우, 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다. 객체 의존 관계를 외부에서 넣어주는 것을 DI(Dependency Injectiono), 의존성 주입이라 한다.
  • 이전 테스트시에는 개발자가 직접 주입했고, 지금은 @Autowired에 의해 스프링이 주입된다.

오류 발생

memberService가 스프링 빈으로 등록되어 있지 않다!!

  • helloController는 스프링이 제공하는 컨트롤러여서 스프링 빈으로 자동 등록된다.
  • @Controller가 있는 경우 자동 등록됨
Spring Bean을 등록하는 2가지 방법
  • 컴포넌트 스캔과 자동 의존관계 설정
  • 자바 코드로 직접 스프링 등록하기

컴포넌트 스캔과 자동 의존관계 설정

  • @Component: 애노테이션이 있으면 스프링 빈으로 자동 등록된다.
  • @Controller: 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.
  • @Component를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록된다.
    (1) @Controller
    (2) @Service
    (3) @Repository

@Service를 더 자세히 봐보기

  • @Component가 선언되어있다.
  • @Controller, @Repository도 마찬가지이다.

회원 서비스 스프링 빈 등록

package hello.hellospring.service;

import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.util.List;
import java.util.Optional;

@Service
public class MemberService {

    private final MemberRepository memberRepository;

    @Autowired
    public MemberService(MemberRepository memberRepository) {
        this.memberRepository = memberRepository;
    }

    //회원가입
    public Long join(Member member){
        validateDuplicateMember(member); // 중복 회원 제거
        memberRepository.save(member);
        return member.getId();
    }

    private void validateDuplicateMember(Member member) { // 중복회원 검증
        memberRepository.findByName(member.getName())// null 가능성이 있으면, Optional 사용
            .ifPresent(m -> {
                throw new IllegalStateException("이미 존재하는 회원입니다");
            });
    }

    public List<Member> findMembers() { // 전체 회원 조회
        return memberRepository.findAll();
    }

    public Optional<Member> findOne(Long memberId){ // 한 회원 조회
        return memberRepository.findById(memberId);
    }
}
  • 생성자에 @Autowired를 사용하면 객체 생성 시점에 스프링 컨테이너에 해당 스프링 빈을 찾아서 주입한다.
  • 생성자 1개만 있는 경우 @Autowired는 생략할 수 있다.

회원 리포지토리 스프링 빈 등록

package hello.hellospring.repository;

import hello.hellospring.domain.Member;
import org.springframework.stereotype.Repository;

import java.util.*;

@Repository
public class MemoryMemberRepository implements MemberRepository{
    private static Map<Long, Member> store = new HashMap<>(); // store 공유
    private static long sequence = 0L;

    @Override
    public Member save(Member member) {
        member.setId(++sequence);
        store.put(member.getId(), member);
        return member;
    }

    @Override
    public Optional<Member> findById(Long id) {
        return Optional.ofNullable(store.get(id));
    }

    @Override
    public Optional<Member> findByName(String name) {
        return store.values().stream() // 일치하는 이름을 찾아서 필터링
                .filter(member -> member.getName() == name)
                .findAny();
    }

    @Override
    public List<Member> findAll() {
        return new ArrayList<>(store.values());
    }

    public void clearStore(){
        store.clear();
    }
}

스프링 빈 등록 이미지

  • memberService와 memberRepository가 스프링 컨테이너에 스프링 빈으로 등록되었다.

    참고 : 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤으로 등록한다.(유일하게 하나만 등록해서 공유한다.)따라서 같은 스프링 빈이면 모두 같은 인스턴스다. 설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용한다.

자바 코드로 직접 스프링 빈 등록하기

  • @Service, @Repository 애노테이션을 제거하고 진행한다.
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();
    }
}
  • 향후 메모리 리포지토리를 다른 리포지토리로 변경할 예정이므로, 컴포넌트 스캔 방식 대신에 자바 코드로 스프링 빈을 설정

참고 : XML로 설정하는 방식도 있지만, 최근에는 잘 사용하지 않으므로 생략한다.

참고 : DI에는 필드 주입, setter 주입, 생성자 주입 이렇게 3가지 방법이 있다. 의존관계가 실행중에 동적으로 변하는 경우는 거의 없으므로 생성자 주입을 권장한다.

참고 : 실무에서는 주로 정형화된 컨트롤러 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용한다. 그리고 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다.

주의 : @Autowired를 통한 DI는 helloController, MemberService등과 같이 스프링이 관리하는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.

0개의 댓글