해당 내용은 김영한씨의 강의 및 스프링 프레임워크 공식 문서, 공식문서 번역을 참고하여 작성한 글입니다.
올바르지 않은 내용이 있다면 의견 남겨주세요. 언제든지 환영입니다.
1. 프론트 컨트롤러
1-1. VER.1
![]() |
![]() |
MVC 패턴을 적용하였지만, 하나의 컨트롤러에 공통 + 비즈니스 로직이 같이 존재한다. 하지만 프론트 컨트롤러 패턴을 사용하면 웹 애플리케이션의 구조를 단순화하고, 공통된 처리 로직을 중앙 집중화할 수 있다. 스프링 MVC는 dispatcherServlet이 프론트컨트롤러 패턴으로 구현되어있다.
공통을 집중한 프론트 컨트롤러 패턴은 다음과 같은 장점이 있다
- 단일 진입점에서 처리하여 공통된 처리 로직을 중앙 집중화할 수 있으므로 코드의 일관성을 유지하고, 중복된 코드를 줄일 수 있다.
- 코드의 유지보수가 수월하다.
1-2. VER.1 의도
- 이전 구조 최대한 유지
- 프론트 컨트롤러는 오직 URL 매핑 정보를 확인 후 해당 URL과 상응하는 컨트롤러를 실행
1-3. VER.1 코드
package hello.servlet.web.frontcontroller.v1;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
public interface ControllerV1 {
/* 예외는 서블렛이랑 동일하게 한 것 */
void process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException;
}
package hello.servlet.web.frontcontroller.v1.controller;
import hello.servlet.domain.member.Member;
import hello.servlet.domain.member.MemberRepository;
import hello.servlet.web.frontcontroller.v1.ControllerV1;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
public class MemberSaveControllerV1 implements ControllerV1 {
private MemberRepository memberRepository = MemberRepository.getInstance();
@Override
public void process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
Member savedMember = new Member();
savedMember.setUsername(request.getParameter("username"));
savedMember.setAge(Integer.parseInt(request.getParameter("age")));
memberRepository.save(savedMember);
request.setAttribute("member", savedMember);
String viewPath = "/WEB-INF/views/save-result.jsp";
request.getRequestDispatcher(viewPath).forward(request, response);
}
}
package hello.servlet.web.frontcontroller.v1;
import hello.servlet.web.frontcontroller.v1.controller.MemberFormControllerV1;
import hello.servlet.web.frontcontroller.v1.controller.MemberListControllerV1;
import hello.servlet.web.frontcontroller.v1.controller.MemberSaveControllerV1;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
/* * 하위에 있는 다 들어옴*/
@WebServlet(name = "frontControllerServletV1", urlPatterns = "/front-controller/v1/*")
public class FrontControllerServletV1 extends HttpServlet {
/* 해당 url 호출하면 FrontControllerServletV1 이 호출 되야 하므로 */
Map<String, ControllerV1> controllerV1Map = new HashMap<String, ControllerV1>();
/*
* 1. 해당 컨트롤러 호출
* 2. 생성자 > url 주소에 해당하는 Controller 저장
* 2-1. controllerV1Map의 url 주소와 일치하는 ControllerV1이 실행 */
public FrontControllerServletV1() {
controllerV1Map.put("/front-controller/v1/members/new-form", new MemberFormControllerV1());
controllerV1Map.put("/front-controller/v1/members/save", new MemberSaveControllerV1());
controllerV1Map.put("/front-controller/v1/members", new MemberListControllerV1());
}
@Override
protected void service(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {
System.out.println("========= FrontControllerServletV1 ==============");
// 생성자에 일치하는 URL 찾기
/*ControllerV1 controller = controllerV1Map.get(requestURI) 는 곧
* ControllerV1 controller = new MemberFormControllerV1() 와 같다 (url이 /front-controller/v1/members/new-form 일 경우) => 다형성 */
String requestURI = req.getRequestURI();
ControllerV1 controller = controllerV1Map.get(requestURI);
// 만약 없으면
if (controller == null) {
res.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
controller.process(req, res);
}
}
- URL 주소를 '/front-controller/v1/*' 로 설정하여 주소가 '/front-controller/v1/save', '/front-controller/v1/list/123' 등 '/front-controller/v1/' 로만 시작한다면 FrontControllerServletV1이 실행된다.
- 다형성을 이용한 controllerV1Map 생성자에서 url 주소에 해당하는 Controller를 찾는다.
💡1-4. 다형성
같은 자료형에 여러가지 타입의 데이터를 대입하여 다양한 결과를 얻어낼 수 있다.
여기서는 MemberSaveControllerV1, MemberListControllerV1, MemberFormControllerV1가 ControllerV1 인터페이스를 상속받고 있다.
참고 : https://reakwon.tistory.com/48
[JAVA] 자바 다형성(Polymorphism) 개념부터 응용 쉬운 설명
다형성(Polymorphism) 다형성이라는 개념은 OOP에서 아주 중요한 개념이므로 모르면 OOP에 대해서 제대로 안다고 할 수 없는 개념입니다. 각 요소들이 여러 가지 자료형으로 표현될 수 있다는 것을 말
reakwon.tistory.com
2. VER2
2-1. VER.2 의도
- 모든 컨트롤러에서 뷰로 이동하는 부분을 중복 제거
String viewPath = "/WEB-INF/views/XXX.jsp";
request.getRequestDispatcher(viewPath).forward(request, response);
2-2. VER.2 코드
package hello.servlet.web.frontcontroller.v2;
import hello.servlet.web.frontcontroller.MyView;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
public interface ControllerV2 {
/* controllerv1과 같은데 반환만 myview로 진행 */
MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException;
}
package hello.servlet.web.frontcontroller.v2.controller;
import hello.servlet.domain.member.Member;
import hello.servlet.domain.member.MemberRepository;
import hello.servlet.web.frontcontroller.MyView;
import hello.servlet.web.frontcontroller.v2.ControllerV2;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
public class MemberSaveControllerV2 implements ControllerV2 {
private MemberRepository memberRepository = MemberRepository.getInstance();
@Override
public MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
Member savedMember = new Member();
savedMember.setUsername(request.getParameter("username"));
savedMember.setAge(Integer.parseInt(request.getParameter("age")));
memberRepository.save(savedMember);
request.setAttribute("member", savedMember);
return new MyView("/WEB-INF/views/save-result.jsp");
}
}
package hello.servlet.web.frontcontroller.v2;
import hello.servlet.web.frontcontroller.MyView;
import hello.servlet.web.frontcontroller.v1.ControllerV1;
import hello.servlet.web.frontcontroller.v2.controller.MemberFormControllerV2;
import hello.servlet.web.frontcontroller.v2.controller.MemberListControllerV2;
import hello.servlet.web.frontcontroller.v2.controller.MemberSaveControllerV2;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
@WebServlet(name = "frontControllerServletV2", urlPatterns = "/front-controller/v2/*")
public class FrontControllerServletV2 extends HttpServlet {
Map<String, ControllerV2> controllerV2Map = new HashMap<String, ControllerV2>();
public FrontControllerServletV2() {
controllerV2Map.put("/front-controller/v2/members/new-form", new MemberFormControllerV2());
controllerV2Map.put("/front-controller/v2/members/save", new MemberSaveControllerV2());
controllerV2Map.put("/front-controller/v2/members", new MemberListControllerV2());
}
@Override
protected void service(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {
System.out.println("========= FrontControllerServletV2 ==============");
String requestURI = req.getRequestURI();
ControllerV2 controller = controllerV2Map.get(requestURI);
// 만약 없으면
if (controller == null) {
res.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
MyView view = controller.process(req, res);
view.render(req, res);
}
}
3. VER3
3-1. VER.3 의도
- 서블릿 종속성을 제거
- Controller에서 사용되지 않는 HttpServletRequest, HttpServletResponse를 개선
- Request를 Model로 사용하는 것이 아닌 별도의 Model 객체 만듬
- 뷰 이름 중복 제거
- 반복되던 ' /WEB-INF/views/' 를 제거
3-2. VER.3 코드
package hello.servlet.web.frontcontroller;
import java.util.HashMap;
import java.util.Map;
public class ModelView {
private String viewName;
private Map<String, Object> model = new HashMap<String, Object>();
public ModelView(String viewName) {
this.viewName = viewName;
}
public String getViewName() {
return viewName;
}
public void setViewName(String viewName) {
this.viewName = viewName;
}
public Map<String, Object> getModel() {
return model;
}
public void setModel(Map<String, Object> model) {
this.model = model;
}
}
package hello.servlet.web.frontcontroller.v3;
import hello.servlet.web.frontcontroller.ModelView;
import java.util.Map;
public interface ControllerV3 {
ModelView process(Map<String, String> paramMap);
}
package hello.servlet.web.frontcontroller.v3.controller;
import hello.servlet.domain.member.Member;
import hello.servlet.domain.member.MemberRepository;
import hello.servlet.web.frontcontroller.ModelView;
import hello.servlet.web.frontcontroller.MyView;
import hello.servlet.web.frontcontroller.v3.ControllerV3;
import java.util.HashMap;
import java.util.Map;
public class MemberSaveControllerV3 implements ControllerV3 {
private MemberRepository memberRepository = MemberRepository.getInstance();
@Override
public ModelView process(Map<String, String> paramMap) {
// paramMap을 통해 데이터 가져오기
String username = paramMap.get("username");
int age = Integer.parseInt(paramMap.get("age"));
// 비즈니스 로직
Member savedMember = new Member();
savedMember.setUsername(username);
savedMember.setAge(age);
memberRepository.save(savedMember);
// model에 담기
ModelView modelView = new ModelView("save-result");
modelView.getModel().put("member", savedMember);
return modelView;
}
}
package hello.servlet.web.frontcontroller.v3;
import hello.servlet.web.frontcontroller.ModelView;
import hello.servlet.web.frontcontroller.MyView;
import hello.servlet.web.frontcontroller.v2.ControllerV2;
import hello.servlet.web.frontcontroller.v2.controller.MemberFormControllerV2;
import hello.servlet.web.frontcontroller.v2.controller.MemberListControllerV2;
import hello.servlet.web.frontcontroller.v2.controller.MemberSaveControllerV2;
import hello.servlet.web.frontcontroller.v3.controller.MemberFormControllerV3;
import hello.servlet.web.frontcontroller.v3.controller.MemberListControllerV3;
import hello.servlet.web.frontcontroller.v3.controller.MemberSaveControllerV3;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
@WebServlet(name = "frontControllerServletV3", urlPatterns = "/front-controller/v3/*")
public class FrontControllerServletV3 extends HttpServlet {
private Map<String, ControllerV3> controllerMap = new HashMap<>();
public FrontControllerServletV3() {
controllerMap.put("/front-controller/v3/members/new-form", new MemberFormControllerV3());
controllerMap.put("/front-controller/v3/members/save", new MemberSaveControllerV3());
controllerMap.put("/front-controller/v3/members", new MemberListControllerV3());
}
@Override
protected void service(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {
System.out.println("========= FrontControllerServletV3 ==============");
String requestURI = req.getRequestURI();
ControllerV3 controller = controllerMap.get(requestURI);
// 만약 없으면
if (controller == null) {
res.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
// 파라미터 전달
Map<String, String> paramMap = createParamMap(req);
ModelView view = controller.process(paramMap);
// 이제 논리 이름 'new-form' 을 /WEB-INF/views/new-form.jsp로 만들어주어야 한다.
// 뷰를 해결해준다는 의미로 viewResolver로 이름 지었다.
MyView myView = viewResolver(view);
// 전달받은 model을 화면에 내려줘야 하니까
myView.render(view.getModel(), req, res);
}
private static MyView viewResolver(ModelView view) {
return new MyView("/WEB-INF/views/" + view.getViewName() + ".jsp");
}
private static Map<String, String> createParamMap(HttpServletRequest req) {
Map<String, String> paramMap = new HashMap<>();
req.getParameterNames().asIterator().forEachRemaining(paramNames -> paramMap.put(paramNames, req.getParameter(paramNames)));
return paramMap;
}
}
4. VER4
4-1. VER.4 의도
- 개발자의 편의성을 증대
return new ModelView("new-form") 로 객체를 생성/반환하지 않도록 수정
4-2. VER.4 코드
package hello.servlet.web.frontcontroller.v4;
import java.util.Map;
public interface ControllerV4 {
String process(Map<String, String> paramMap, Map<String, Object> model);
}
package hello.servlet.web.frontcontroller.v4.controller;
import hello.servlet.domain.member.Member;
import hello.servlet.domain.member.MemberRepository;
import hello.servlet.web.frontcontroller.ModelView;
import hello.servlet.web.frontcontroller.v3.ControllerV3;
import hello.servlet.web.frontcontroller.v4.ControllerV4;
import java.util.Map;
public class MemberSaveControllerV4 implements ControllerV4 {
private MemberRepository memberRepository = MemberRepository.getInstance();
@Override
public String process(Map<String, String> paramMap, Map<String, Object> model) {
// paramMap을 통해 데이터 가져오기
String username = paramMap.get("username");
int age = Integer.parseInt(paramMap.get("age"));
// 비즈니스 로직
Member savedMember = new Member();
savedMember.setUsername(username);
savedMember.setAge(age);
memberRepository.save(savedMember);
// 모델 담기
model.put("member", savedMember);
return "save-result";
}
}
package hello.servlet.web.frontcontroller.v4;
import hello.servlet.web.frontcontroller.ModelView;
import hello.servlet.web.frontcontroller.MyView;
import hello.servlet.web.frontcontroller.v3.ControllerV3;
import hello.servlet.web.frontcontroller.v3.controller.MemberFormControllerV3;
import hello.servlet.web.frontcontroller.v3.controller.MemberListControllerV3;
import hello.servlet.web.frontcontroller.v3.controller.MemberSaveControllerV3;
import hello.servlet.web.frontcontroller.v4.controller.MemberFormControllerV4;
import hello.servlet.web.frontcontroller.v4.controller.MemberListControllerV4;
import hello.servlet.web.frontcontroller.v4.controller.MemberSaveControllerV4;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
@WebServlet(name = "frontControllerServletV4", urlPatterns = "/front-controller/v4/*")
public class FrontControllerServletV4 extends HttpServlet {
private Map<String, ControllerV4> controllerMap = new HashMap<>();
public FrontControllerServletV4() {
controllerMap.put("/front-controller/v4/members/new-form", new MemberFormControllerV4());
controllerMap.put("/front-controller/v4/members/save", new MemberSaveControllerV4());
controllerMap.put("/front-controller/v4/members", new MemberListControllerV4());
}
@Override
protected void service(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {
System.out.println("========= FrontControllerServletV4 ==============");
String requestURI = req.getRequestURI();
ControllerV4 controller = controllerMap.get(requestURI);
// 만약 없으면
if (controller == null) {
res.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
// 파라미터 전달
Map<String, String> paramMap = createParamMap(req);
Map<String, Object> model = new HashMap<>();
String view = controller.process(paramMap, model);
MyView myView = viewResolver(view);
// 전달받은 model을 화면에 내려줘야 하니까
myView.render(model, req, res);
}
private static MyView viewResolver(String view) {
return new MyView("/WEB-INF/views/" + view + ".jsp");
}
private static Map<String, String> createParamMap(HttpServletRequest req) {
Map<String, String> paramMap = new HashMap<>();
req.getParameterNames().asIterator().forEachRemaining(paramNames -> paramMap.put(paramNames, req.getParameter(paramNames)));
return paramMap;
}
}
5. VER5
5-1. VER.5 의도
- 현재 프론트 컨트롤러는 한가지 방식의 프론트 컨트롤러 인터페이스만 사용할 수 있어 다양한 방식의 컨트롤러 인터페이스를 호환할 수 있도록 개선
5-2. VER.5 코드
package hello.servlet.web.frontcontroller.v5;
import hello.servlet.web.frontcontroller.ModelView;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
public interface MyHandlerAdapter {
// 컨트롤러 넘어왔을때, 나 이 컨트롤러 지원 가능해!
boolean supports(Object handler);
ModelView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException;
}
package hello.servlet.web.frontcontroller.v5;
import hello.servlet.web.frontcontroller.ModelView;
import hello.servlet.web.frontcontroller.MyView;
import hello.servlet.web.frontcontroller.v3.controller.MemberFormControllerV3;
import hello.servlet.web.frontcontroller.v3.controller.MemberListControllerV3;
import hello.servlet.web.frontcontroller.v3.controller.MemberSaveControllerV3;
import hello.servlet.web.frontcontroller.v4.ControllerV4;
import hello.servlet.web.frontcontroller.v4.controller.MemberFormControllerV4;
import hello.servlet.web.frontcontroller.v4.controller.MemberListControllerV4;
import hello.servlet.web.frontcontroller.v4.controller.MemberSaveControllerV4;
import hello.servlet.web.frontcontroller.v5.adapter.ControllerV3HandlerAdapter;
import hello.servlet.web.frontcontroller.v5.adapter.ControllerV4HandlerAdapter;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ViewResolver;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@WebServlet(name = "frontControllerServletV5", urlPatterns = "/front-controller/v5/*")
public class FrontControllerServletV5 extends HttpServlet {
// 아무 컨트롤러 다 들어갈 수 있어야 하므로
private final Map<String, Object> handlerMappingMap = new HashMap<String, Object>();
private final List<MyHandlerAdapter> handlerAdapters = new ArrayList<MyHandlerAdapter>();
public FrontControllerServletV5() {
initHandlerMappingMap();
initHandlerAdapters();
}
private void initHandlerAdapters() {
handlerAdapters.add(new ControllerV3HandlerAdapter());
handlerAdapters.add(new ControllerV4HandlerAdapter());
}
private void initHandlerMappingMap() {
handlerMappingMap.put("/front-controller/v5/v3/members/new-form", new MemberFormControllerV3());
handlerMappingMap.put("/front-controller/v5/v3/members/save", new MemberSaveControllerV3());
handlerMappingMap.put("/front-controller/v5/v3/members", new MemberListControllerV3());
//V4 추가
handlerMappingMap.put("/front-controller/v5/v4/members/new-form", new MemberFormControllerV4());
handlerMappingMap.put("/front-controller/v5/v4/members/save", new MemberSaveControllerV4());
handlerMappingMap.put("/front-controller/v5/v4/members", new MemberListControllerV4());
}
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("========= FrontControllerServletV5 ==============");
Object handler = getHandler(request);
if (handler == null) {
response.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
MyHandlerAdapter adapter = getHandlerAdapter(handler);
ModelView mv = adapter.handle(request, response, handler);
MyView view = viewResolver(mv.getViewName());
view.render(mv.getModel(), request, response);
}
private Object getHandler(HttpServletRequest request) {
String requestURI = request.getRequestURI();
return handlerMappingMap.get(requestURI);
}
private MyHandlerAdapter getHandlerAdapter(Object handler) {
for (MyHandlerAdapter adapter : handlerAdapters) {
if (adapter.supports(handler)) {
return adapter;
}
}
throw new IllegalArgumentException("handler adapter를 찾을 수 없습니다. handler=" + handler);
}
private MyView viewResolver(String viewName) {
return new MyView("/WEB-INF/views/" + viewName + ".jsp");
}
}
package hello.servlet.web.frontcontroller.v5.adapter;
import hello.servlet.web.frontcontroller.ModelView;
import hello.servlet.web.frontcontroller.v3.ControllerV3;
import hello.servlet.web.frontcontroller.v4.ControllerV4;
import hello.servlet.web.frontcontroller.v5.MyHandlerAdapter;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
// v3를 지원하는 어댑터
public class ControllerV4HandlerAdapter implements MyHandlerAdapter {
@Override
public boolean supports(Object handler) {
return (handler instanceof ControllerV4); // V3가 맞으면 true
}
@Override
public ModelView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException {
ControllerV4 controller = (ControllerV4) handler; // cast
// 파라미터 세팅
Map<String, String> paramMap = createParamMap(request);
Map<String, Object> model = new HashMap<>();
String viewName = controller.process(paramMap, model);
ModelView modelView = new ModelView(viewName);
modelView.setModel(model);
return modelView;
}
private static Map<String, String> createParamMap(HttpServletRequest req) {
Map<String, String> paramMap = new HashMap<>();
req.getParameterNames().asIterator().forEachRemaining(paramNames -> paramMap.put(paramNames, req.getParameter(paramNames)));
return paramMap;
}
}
1) 참고 - 프론트 컨트롤러
https://f-lab.kr/insight/front-controller-pattern-20240912
효율적인 웹 애플리케이션 개발을 위한 프론트 컨트롤러 패턴
이 블로그 포스트는 프론트 컨트롤러 패턴의 개요와 동작 원리, 예제, 장점과 단점에 대해 다룹니다. 프론트 컨트롤러 패턴을 통해 웹 애플리케이션의 구조를 단순화하고, 코드의 유지보수성을
f-lab.kr
'Java > Spring' 카테고리의 다른 글
[Spring] JAVA 1.8 gradle 세팅 (1) | 2024.12.18 |
---|---|
[Spring] MVC 발전과정(1) (0) | 2024.10.30 |
[Spring] 빈(Bean) 생명주기 (0) | 2023.02.23 |
[Spring] Singleton (0) | 2023.02.03 |
[Spring] IOC (0) | 2023.01.26 |