반응형

# 스프링 빈과 의존관계

## 스프링 빈 등록 방법 (2가지 존재)

  • 스프링은 스프링 컨테이너에 스프링 빈 등록 시 기본적으로 싱글 톤 (유일하게 하나만 등록하여 공유) 으로 등록. (설정으로 싱글 톤이 아니도록 할 수 있지만, 대부분 싱글톤을 사용한다.)
  1. 컴포넌트 스캔과 자동 의존관계 설정.
  2. 자바 코드로 직접스프링 빈 등록.

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

@Controller

  • @Controller 라는 에노테이션 존재 시 스프링 빈으로 자동 등록되어 스프링 컨테이너에서 관리하게 됨.
  • 서비스의 경우 @Service 사용.
  • 리포지토리의 경우 @Repository 사용.

@Autowired

  • 생성자에 @Autowired 라는 에노테이션 존재 시 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다. (의존성 주입 : DI)

의존성 주입 (DI)

  • @Autowired 처럼 객체 의존관계를 외부에서 넣어주는 것.

스프링 빈 등록

  • helloController -> memberService -> memberRepository
  • @Controller & @Autowired
@Controller
public class MemberController {

    private final MemberService memberService;

    @Autowired
    public MemberController(MemberService memberService) {
        this.memberService = memberService;
    }

}
  • @Service & Autowired
@Service
public class MemberService {

    private final MemberRepository memberRepository;

    @Autowired
    public MemberService(MemberRepository memberRepository) {
        this.memberRepository = memberRepository;
    }
    ...
}
  • @Repository
@Repository
public class MemoryMemberRepository implements MemberRepository {

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

  • @Component 에노테이션이 있으면 스프링 빈으로 자동 등록된다. @Controller 에노테이션이 있으면 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문임.
  • @Component 를 포함하는 에노테이션도 스프링 빈으로 자동 등록된다. (@Controller, @Service, @Repository)
  • @Autowired 의 경우 연관관계를 연결해주는 역할임. (helloController -> memberService -> memberRepository 연결)

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

2. 자바 코드로 직접 스프링 빈 등록.

  • SpringConfig 생성 후 @Configuration 에노테이션 적용.
  • SpringConfig
  • @Autowired를 통한 DI의 경우 스프링이 관리하는 객체에서만 동작한다. (아래와 같이 스프링 빈으로 등록한 경우)
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(); // 구현체
    }
}
  • MemberController
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;
    }

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

의존성 주입 방법 3가지.

1. 생성자 주입 (권장)

@Autowired
public MemberController(MemberService memberService) {
    this.memberService = memberService;
}

2. 필드 주입 (비추)

@Autowired private final MemberService memberService;

3. Setter 주입 (비추)

  • 단점 : 아래와 같은 경우 누군가가 MemberController 호출 시 public 으로 열려있어야 함.
@Autowired
public void setMemberService(MemberService memberService) {
    this.memberService = memberService;
}

 

반응형
반응형

# 스프링 빈과 의존관계

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

  • @Autowired : 생성자에 @Autowired 가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다. (DI (Dependency Injection = 의존관계를 외부에서 넣어주는 것 = 의존성 주입)).
  • 기존 소스코드 오류 제거를 위해 어노테이션 추가. (@Controller / @Service / @Repository / @Autowired)
  • 스프링 빈을 등록하는 2가지 방법.
- 컴포넌트 스캔과 자동 의존관계 설정
(컴포넌트 스캔 : @Controller / @Service / @Repository / @Autowired)

- 자바 코드로 직접 스프링 빈 등록하기
  • 컴포넌트 스캔 원리
- @Component 애노테이션이 있으면 스프링 빈으로 자동 등록 됨. 

- @Controller 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문.
  • @Component 를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록 됨.
- @Controller 

- @Service

- @Repository
  • 생성자에 @Autowired 사용 시 객체 생성 시점에 스프링 컨테이너에서 해당 스프링 빈을 찾아서 주입. (생성자가 1개만 있는 경우 @Autowired 생략 가능.)
  • 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때 기본으로 싱글톤(유일하게 하나만 등록해서 공유)으로 등록 함. 따라서 같은 스프링 빈이면 모두 같은 인스턴스이다. 설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용.

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

  • 기존 회원 서비스와 회원 리포지토리의 @Service, @Repository, @Autowired 애노테이션을 제거하고 SpringConfig 생성하여 진행.
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();
    }
}
  • DI 방법 : 의존관계가 실행 중 동적으로 변하는 일은 없으므로 생성자 주입 방법 권장.
1. 생성자 주입 (권장)
@Autowired
public MemberController(MemberService memberService) {
	this.memberService = memberService;
}


2. 필드 주입
@Autowired private MemberService memberService;


3. setter 주입
@Autowired
public void setMemberService(MemberService memberService) {
	this.memberService = memberService;
}
  • 실무에서는 주로 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용.
  • 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록.
  • @Autowired 를 통한 DI는 helloConroller , memberService 등과 같이 스프링이 관리하는 객체에서만 동작. 스프링 빈으로 등록하지 않고, 직접 생성한 객체에서는 동작하지 않는다.

 

반응형
반응형

# 스프링 빈과 의존관계

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

  • 생성자에 @Autowired 가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다.
  • 객체 의존관계를 외부에서 넣어주는 것을 DI (Dependency Injection), 의존성 주입이라고 한다.
  • 이전 테스트에서는 개발자가 직접 주입했고, 여기서는 @Autowired에 의해 스프링이 주입해준다.
  • 오류발생
Consider defining a bean of type 'hello.hellospring.service.MemberService' in
your configuration.

  • 참고 : helloController는 스프링이 제공하는 컨트롤러여서 스프링 빈으로 자동 등록된다.
  • @Controller 가 있으면 자동 등록된다.

## 스프링 빈을 등록하는 방법 (2가지 존재)

  • 컴포넌트 스캔과 자동 의존관계 설정
  • 자바 코드로 직접 스프링 빈 등록하기

## 컴포넌트 스캔 원리

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

## 회원 서비스_스프링 빈 등록

@Service
public class MemberService {
   
   private final MemberRepository memberRepository;
   
   @Autowired
   public MemberService(MemberRepository memberRepository) {
		this.memberRepository = memberRepository;
   }
}
  • 참고 : 생성자에 @Autowired 를 사용하면 객체 생성 시점에 스프링 컨테이너에서 해당 스프링 빈을 찾아서 주입한다. 생성자가 1개만 있으면 @Autowired 는 생략할 수 있다.

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

@Repository
public class MemoryMemberRepository implements MemberRepository {}

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

 

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

  • 자바 코드로 직접 스프링 빈을 등록하기 위해 기존의 회원 서비스와 회원 리포지토리의 @Service, @Repository, @Autowired 애노테이션을 제거하고 진행.
  • SpringConfig
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는 helloConroller , memberService 등과 같이 스프링이 관리하는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.

 

# 회원 웹 기능__홈 화면 추가

  • HomeController 추가
package hello.hellospring.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;

@Controller
public class HomeController {

    @GetMapping("/")
    public String home() {
        return "home";
    }

}
  • home.html 회원 관리용 홈 추가
<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<body>

<div class="container">
    <div>
        <h1>Hello Spring</h1>
        <p>회원 기능</p>
        <p>
            <a href="/members/new">회원 가입</a>
            <a href="/members">회원 목록</a>
        </p>
    </div>
</div><!-- /container -->

</body>
</html>
  • 참고 : 컨트롤러가 정적 파일보다 우선순위가 높다

 

#  회원 웹 기능_등록

  • MemberController 회원 등록 폼 컨트롤러
package hello.hellospring.controller;

import hello.hellospring.domain.Member;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;

@Controller
public class MemberController {

    private final MemberService memberService;

    @Autowired
    public MemberController(MemberService memberService) {
        this.memberService = memberService;
    }

    @GetMapping("/members/new")
    public String createForm() {
        return "members/createMemberForm";
    }

    @PostMapping("/members/new")
    public String create(MemberForm form) {
        Member member = new Member();
        member.setName(form.getName());

        memberService.join(member);

        return "redirect:/";
    }
}
  • createMemberForm.html 회원 등록 폼
<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<body>
<div class="container">
    <form action="/members/new" method="post">
        <div class="form-group">
            <label for="name">이름</label>
            <input type="text" id="name" name="name" placeholder="이름을 입력하세요">
        </div>
        <button type="submit">등록</button>
    </form>
</div> <!-- /container -->
</body>
</html>
  • MemberForm 회원 컨트롤러에서 회원을 실제 등록하는 기능
package hello.hellospring.controller;

public class MemberForm {
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}
  • GET : 조회 시 사용
  • POST : 데이터 등록 시 사용

 

# 회원 웹 기능_조회

  • MemberController 회원 컨트롤러에서 조회기능
package hello.hellospring.controller;

import hello.hellospring.domain.Member;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;

import java.util.List;

@Controller
public class MemberController {

    private final MemberService memberService;

    @Autowired
    public MemberController(MemberService memberService) {
        this.memberService = memberService;
    }

    @GetMapping("/members/new")
    public String createForm() {
        return "members/createMemberForm";
    }

    @PostMapping("/members/new")
    public String create(MemberForm form) {
        Member member = new Member();
        member.setName(form.getName());

        memberService.join(member);

        return "redirect:/";
    }

    @GetMapping("/members")
    public String list(Model model) {
        List<Member> members = memberService.findMembers();
        model.addAttribute("members", members);

        return "members/memberList";
    }
}
  •  memberList 회원 리스트
<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<body>
<div class="container">
    <div>
        <table>
            <thead>
            <tr>
                <th>#</th>
                <th>이름</th>
            </tr>
            </thead>
            <tbody>
            <tr th:each="member : ${members}">
                <td th:text="${member.id}"></td>
                <td th:text="${member.name}"></td>
            </tr>
            </tbody>
        </table>
    </div>
</div> <!-- /container -->
</body>
</html>

 

출처 : 인프런 스프링 입문

반응형

+ Recent posts