[스프링 입문] Spring Boot 입문(2)
5. 스프링 빈과 의존 관계
@를 통해서 Annotation을 설정하면 스프링 컨테이너가 스프링 빈을 생성을 해서 관리한다.
package hello.hello_spring.controller;
import hello.hello_spring.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로 어노테이션을 설정해주면 스프링 컨테이너에서 해당 서비스를 가져와서 연결해준다.
대신 MemberService 클래스에 @Service를 통해서 서비스로 등록을 해줘야한다.
레포지토리 역시 @Repository로 설정을 해준다.
스프링 빈을 등록하는 방법에는 아래의 2가지 방법이 있다.
1) 컴포넌트 스캔과 자동 의존관계 설정
2) 자바 코드로 직접 스프링 빈 등록하기
1) 컴포넌트(@Component) 스캔과 자동 의존관계 설정
@Component 어노테이션을 설정하면 자동으로 Spring Bean에 등록된다.
@Component는 @Service, @Controller, @Repository 등등에 포함되어 있다.
스프링이 @Component에 관련되어 있으면 Spring Bean에 등록을 하고 @Autowired는 아래 그림의 화살표라고 생각하면 좋을 거 같다.
생성자에 @Autowired를 사용하면 객체 생성 시점에 스프링 컨테이너에서 해당 스프링 빈을 찾아서 주입한다.
생성자가 하나만 있다면 @Autowired를 생략할 수 있다.
스프링은 스프링 컨테이너에 스프링 빈을 등록할 때 기본으로 싱글톤으로 등록한다.
같은 스프링 빈은 모두 같은 인스턴스이다.
Controller에서 서비스를 생성하고 서비스는 Repository를 생성하는데 이처럼 외부에서 의존성을 주입해주는 것을 DI(Dependency Injection, 의존성 주입)이라고 한다.
2) 자바 코드로 직접 스프링 빈 등록하기
package hello.hello_spring;
import hello.hello_spring.repository.MemberRepository;
import hello.hello_spring.repository.MemoryMemberRepository;
import hello.hello_spring.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();
}
}
따로 SpringConfig 파일을 생성하여 Service에는 Repository를 Controller에서는 Service를 호출해서 사용한다.
그럼에도 @Controller는 그대로 사용을 해준다(위의 Autowired 코드 그대로)
DI(Dependency Injection) 방식
1. 필드 주입
2. setter 주입
3. 생성자 주입
package hello.hello_spring.controller;
import hello.hello_spring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
@Controller
public class MemberController {
private final MemberService memberService;
// 1. 필드 주입
@Autowired private MemberService meberService;
// 2. setter 주입 > 아무나 set 접근할 수 있어서 권장하지 않음
public setMemberController(MemberService memberService) {
this.memberService = memberService;
}
// 3. 생성자 주입
public MemberController(MemberService memberService) {
this.memberService = memberService;
}
}
실행 중에는 의존 관계가 변하는 경우는 거의 없어서 3번 생성자 주입을 권장한다.
실무에서는 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용하고 나머지 경우에는 설정을 통해서 스프링 빈을 등록한다.
@Autowired를 통한 DI는 helloController, MemberService 등과 같이 스프링이 관리하는 객체에서만 동작한다.
6. 웹 MVC 개발
회원 가입하는 Controller를 생성한다.
이름은 HomeController로 하였고 home.html을 return하게 한다.
home.html에서 사용하는 회원 가입(/members/new)과 회원 목록(/members)를 구현하여야 한다.
package hello.hello_spring.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
@Controller
public class HomeController {
@GetMapping("/")
public String home(){
return "home";
}
}
<!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>
먼저 MemberForm class를 생성하여 입력받을 멤버 구조를 설정한다.
package hello.hello_spring.controller;
public class MemberForm {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
이 때 회원 가입 html은 아래와 같으며 "/members/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>
회원 목록 조회 html은 아래와 같으며 "/members/memberList.html"이다.
<!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>
그리고 MemberController에 회원 가입과 회원 목록 조회 기능을 구현한다.
package hello.hello_spring.controller;
import hello.hello_spring.domain.Member;
import hello.hello_spring.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.ArrayList;
import java.util.List;
@Controller
public class MemberController {
@Autowired
private final MemberService memberService;
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";
}
}
create는 멤버를 새로 만드는 것인데 MemberForm.html에서 입력받은 name을 Member name으로 set해주고 서비스를 통해서 로직을 실행한다.
Controller -> Service -> Repository 인 것을 항상 생각하자.
list의 경우는 서비스를 통해서 모든 멤버를 찾아서 List로 만든 후 html을 통해서 값을 보여준다.
해당 서비스는 새로고침을 하면 값이 증발한다.
서버를 다운시키면 값이 사라지는데 해당 값을 저장하기 위해 DB에 연결해야한다.