@Controller ,@Repostory, @Service 는 사실상 @Component이다. 스프링이 컴포넌트 스캔을 쫘악한다. 기본적으로는 HelloSpringApplication.java 맨위에 package hello.hellospring; 하위만 기본으로 스캔을 해준다. 그 위의 상위 패키지들은 따로 설정을 해주어야 한다.
@Autowired 를 통한 DI는 해당 클래스가 스프링 빈으로 등록이 되어있어야지만 작동한다.
스캔을 하면서 생성자를 호출해서 객체를 생성하는데 싱글톤으로 생성해준다.
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Repository {
/**
* The value may indicate a suggestion for a logical component name,
* to be turned into a Spring bean in case of an autodetected component.
* @return the suggested component name, if any (or empty String otherwise)
*/
@AliasFor(annotation = Component.class)
String value() default "";
}
Annotation 말고 직접 자바로 컴포넌트 설정을 해 줄 수 있다. @Configuration 을 써준후 안에 빈으로 등록하고 싶은 클래스들을 적으면 된다. 또한 MemoryMeberRepostory 에서 DBMeberRepository로 쉽게 바꿔줄수 가있다. 컴포넌트 스캔의 경우는 여러 파일들을 변경해주어야 한다. 이것이 직접 설정파일을 운영할 때 가지는 장점이다.
@Configuration
public class SpringConfig
{
@Bean
public MemberService memberService()
{
return new MemberService(memberRepository());
}
@Bean
public MemberRepository memberRepository()
{
return new MemoryMeberRepository();
}
}
Setter injection 보다는 Constructor injection 을 써야한다. 의존관계가 실행중에 동적으로 변하는 경우는 없기 때문이다. 또한 다른놈이 setter method를 호출 할 수 도 있기 때문이다. final 키워드가 걸린 variable 을 생성자 시점까지 초기화를 해주어야 한다.
간단한 Member management
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";
}
}
기존의 static folder 에 있는 index.html 보다 Controller안에 있는 GetMapping 이 더 우선순위를 가진다. 저것을 지우면 다시 index.html 이 렌더링 된다.
<!DOCTPYE 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>
home.html 에는 회원가입을 누르면 /members/new 로 Get request를 보내주고 회원목록을 누르면 /members 로 Get request를 보내준다.
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:/";
}
}
회원가입을 누르면 Getmapping이 그 request를 받아들여서 members/createMembersForm.html 페이지로 이동한다.
<!DOCTPYE 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>
</form>
</div>
</body>
</html>
타임리프 엔진이 이 페이지를 렌더링 해준다. 이떄 form 태그 안에 input name attribute를 name으로 해주고 정보를 입력하면 PostMapping 으로 간후에 인자에 MemberForm 을 확인한다.
package hello.hellospring.controller;
public class MemberForm
{
private String name;
public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}
}
해당되는 SetXxxx 를 찾는다 이 경우에는 SetName함수를 찾아서 호출 하여 MemberForm의 private String name; 을 채워준다.
MemberController 에서 회원 목록을 보기 위한 GetMapping 하나를 추가해준다.
@GetMapping("/members")
public String list(Model model)
{
List<Member> members = memberService.findMembers();
model.addAttribute("members",members);
return "members/memberList";
}
Model을 통해 members key에 members value를 추가해준후에 members/memberList.html 을 함꼐 렌더링 해준다.
<!DOCTPYE 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>
model 에서 member key 에 해당되는 value들을 thymeleaf 방식의 for each 구문을 통해 화면에 렌더링 해준다.
'WEB > Spring' 카테고리의 다른 글
김영한 (스프링 부트 5) (0) | 2021.02.08 |
---|---|
김영한 (스프링부트 입문) 4 (0) | 2021.02.08 |
김영한 (스트링부트 입문) 2 (1) | 2021.01.24 |
김영한 (스프링부트 입문) 1 (0) | 2021.01.24 |
Spring Ehcache example (0) | 2021.01.23 |