Bean

  • 스프링 컨테이너가 관리하는 객체.
  • @Component, @Service, @Repository, @Controller 어노테이션을 통해 자동 등록 가능.
  • 또는 @Configuration으로 선언된 클래스에 @Bean 메소드로 수동 등록 가능. 스프 링이 @Bean이 붙은 메소드를 실행하고, 반환된 객체를 빈으로 등록한다.
  • 기본적으로 싱글톤 스코프로 동작한다.

 

Autowired

  • 객체 간 의존성을 자동으로 주입. 스프링 컨테이너에 등록된 Bean을 자동으로 찾아서 주입해준다.
  • @Autowired가 붙은 생성자를 확인하고, 필요한 MemberRepository가 있는지 찾는다.
  • 스프링 컨테이너에 등록되어 있으면 자동으로 주입해주고, 없으면 예외 발생.
  • 생성자 주입 방식일 때 생성자가 1개만 있으면 생략 가능하다.
  • 스프링이 자동으로 의존 관계에 따라 생성 순서를 조정해서 주입해주기 때문에, 따로 객체 순서를 신경 쓸 필요가 없다.
  • A <-> B 가 서로 필요로 하는 순환 참조 문제가 발생할 때는 @Lazy를 이용하거나, 세터 주입으로 해결해주면 된다.
  • 하지만 순환 참조가 발생하는 설계 자체의 문제가 의심되니, 의존성 방향을 다시 검토하는 것이 좋다.

 

 

홈 컨트롤러

  • 컨트롤러가 정적 파일보다 우선순위가 높다.
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;

@Controller
public class HomeController {

    @GetMapping("/")
    public String home() {
        return "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>
</body>
</html>

 

 

멤버 컨트롤러

// Controller 어노테이션을 통해, 스프링 컨테이너가 뜰 때 해당 객체를 생성해서 관리(스프링 컨테이너가). -> 컴퍼넌트 스캔
@Controller
public class MemberController {

    private final MemberService memberService;

    // 스프링 컨테이너에 해당 MemberService 객체를 연결(단 하나만 생성해서 관리하기 위함 -> 싱글톤)
    // 생성자 주입 방식. 어차피 한 번만 세팅될거라, 굳이 세터 주입이나 필드 주입대신, 생성자 주입 방식이 적절하다.
    @Autowired
    public MemberController(MemberService memberService) {
        this.memberService = memberService;
    }

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

    @PostMapping("/members/new")
    public String create(MemberForm form) {
    	// DTO를 통해 값 세팅
        Member member = new Member();
        member.setName(form.getName());

        memberService.join(member);

		// 회원 가입 끝나면 홈 화면으로 이동
        return "redirect:/";
    }

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

        // 뷰 템플릿에 members 데이터 전송할 수 있도록
        model.addAttribute("members", members);

        return "members/memberList";
    }
}

 

 

멤버 폼

  • 도메인 모델과 입력 데이터를 분리하기 위해, DTO(Data Transfer Object) 역할로 둠. 요청과 응답을 위한 역할일 뿐.
  • Member 클래스는 DB 엔티티로 사용될 수도 있고, 특정 비즈니스 로직이 포함될 수 있기 때문.
public class MemberForm {

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

 

 

회원 생성 및 회원 리스트 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>

</body>
</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>
  </body>
</html>