-서블릿 방식 요청 처리를 학습한다.
-요청 방식(http get 쿼리파라미터, html form 전송, api 전송 등)에 따른 처리 방법을 학습한다.
-목차는 강의 순서대로 진행한다.
-모든 소스는 깃허브에서 관리한다.(https://github.com/coderahn/Spring-Lecture4)
2.서블릿
스프링 MVC 방식을 살펴보기 전에 프레임워크 없이 서블릿 스타일의 클라이언트-서버 통신 과정을 살펴본다.
1.프로젝트 생성
start.spring.io에서 스프링 부트 프로젝트를 생성하고 인텔리제이 프로젝트 오픈 및 설정을 한다. 그리고 롬복도 설치한다. start.spring.io에서 Dependencies로 lombok을 추가하여 생성하자. 그리고 스프링부트는 Jar로 빌드되지만 JSP사용을 위해 War 선택 후 Generate하여 프로젝트를 생성한다.
이후 인텔리제이의 Welcome to IntelliJ IDEA 창에서 Open을 클릭하여 압축을 푼 프로젝트 경로의 build.gradle을 선택하여 연다.
롬복 설치 후 다음과 같이 Settings > Annotation Processors의 Enable annotation processing을 체크한다.
2.Hello 서블릿
서블릿 컴포넌트 스캔을 위한 설정 어노테이션을 다음과 같이 추가한다.
[ServletApplication.java]
package hello.servlet;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.servlet.ServletComponentScan;
@ServletComponentScan //서블릿 자동 등록
@SpringBootApplication
public class ServletApplication {
public static void main(String[] args) {
SpringApplication.run(ServletApplication.class, args);
}
}
다음으로 hello.servlet.basic 패키지 밑에 HelloServlet.java을 생성하여 다음과 같이 코드를 작성한다.
- 클래스에 @WebServlet 선언(urlPattern 설정 가능)
- HttpServlet 상속 : service 메소드 오버라이드
- service 메소드 오버라이드 : urlPattern에 맞는 요청이 오면 자동 실행
[HelloServlet.java]
package hello.servlet.basic;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet(name="helloServlet", urlPatterns = "/hello")
public class HelloServlet extends HttpServlet {
//서블릿 호출시(urlPattern) service 호출
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("HelloServlet.service");
System.out.println("request = " + request); //org.apache.catalina.connector.RequestFacade@7aa3142
System.out.println("response = " + response); //org.apache.catalina.connector.ResponseFacade@290b21c9
//queryString 조회!
String username = request.getParameter("username");
System.out.println("username = " + username);
//header정보에 넣기(response의 content-type)
response.setContentType("text/plain");
response.setCharacterEncoding("utf-8");
//http message body에 데이터가 들어감 -> 화면에 hello + username 보여줌
response.getWriter().write("hello " + username);
}
}
3.HttpServletRequest - 개요
-
HTTP 요청메세지
-
스타트라인 : HTTP메소드, URL, 쿼리스트링, 스키마, 프로토콜
-
헤더 : 헤더정보(Host, Content-Type..)
-
바디 : form파라미터 형식 조회, message body 데이터 직접 조회
-
-
HttpServletRequest는 파싱 기능 외에 부가기능도 있음
-
임시저장소 기능세션 관리 기능
-
저장 : request.setAttribute(name, value)
-
조회 : request.getAttribute(name)
-
-
세션 관리 기능
- request.getSession(create : true)
-
4.HttpServletRequest - 기본 사용법
Header정보 및 스타트라인 정보 등을 꺼내는 방법을 알아본다.
Http 요청 구성에서 스타트라인은 다음과 같은 메소드들로 확인할 수 있다.
[RequestHeaderServlet.java]
@WebServlet(name = "requestHeaderServlet", urlPatterns = "/request-header")
public class RequestHeaderServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
printStartLine(request);
printHeaders(request);
printHeaderUtils(request);
printEtc(request);
}
//HTTP - 스타트라인 출력
private void printStartLine(HttpServletRequest request) {
System.out.println("--- REQUEST-LINE - start ---");
System.out.println("request.getMethod() = " + request.getMethod()); //GET
System.out.println("request.getProtocol() = " + request.getProtocol()); //HTTP/1.1
System.out.println("request.getScheme() = " + request.getScheme()); //http
// http://localhost:8080/request-header
System.out.println("request.getRequestURL() = " + request.getRequestURL());
// /request-test
System.out.println("request.getRequestURI() = " + request.getRequestURI());
//username=hi
System.out.println("request.getQueryString() = " + request.getQueryString());
System.out.println("request.isSecure() = " + request.isSecure()); //https 사용 유무
System.out.println("--- REQUEST-LINE - end ---");
System.out.println();
}
//...
}
Http Header정보는 다음과 같이 확인할 수 있다.
[RequestHeaderServlet.java]
@WebServlet(name = "requestHeaderServlet", urlPatterns = "/request-header")
public class RequestHeaderServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
printStartLine(request);
printHeaders(request);
printHeaderUtils(request);
printEtc(request);
}
//HTTP - 스타트라인 출력
//...
//HTTP - Header 모든 정보
private void printHeaders(HttpServletRequest request) {
System.out.println("--- Headers - start ---");
//예전방법
// Enumeration<String> headerNames = request.getHeaderNames();
// while(headerNames.hasMoreElements()) {
// String headerName = headerNames.nextElement();
// String value = request.getHeader(headerName);
// System.out.println(headerName + ": " + value);
// }
//요즘방법
request.getHeaderNames().asIterator().
forEachRemaining(headerName -> System.out.println(headerName + ": " + request.getHeader(headerName)));
System.out.println("--- Headers - end ---");
System.out.println();
}
}
[RequestHeaderServlet.java]
@WebServlet(name = "requestHeaderServlet", urlPatterns = "/request-header")
public class RequestHeaderServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
printStartLine(request);
printHeaders(request);
printHeaderUtils(request);
printEtc(request);
}
//HTTP - 스타트라인 출력
//...
//HTTP - Header 모든 정보
//...
//Header 편리한 조회
private void printHeaderUtils(HttpServletRequest request) {
System.out.println("--- Header 편의 조회 start ---");
System.out.println("[Host 편의 조회]");
System.out.println("request.getServerName() = " + request.getServerName()); //Host 헤더
System.out.println("request.getServerPort() = " + request.getServerPort()); //Host Port
System.out.println();
System.out.println("[Accept-Language 편의 조회]");
request.getLocales().asIterator().forEachRemaining(locale -> System.out.println("locale = " + locale));
System.out.println("request.getLocale() = " + request.getLocale()); //우서순위값 높은 거 꺼냄(ko)
System.out.println();
System.out.println("[cookie 편의 조회]");
if (request.getCookies() != null) {
for (Cookie cookie : request.getCookies()) {
System.out.println(cookie.getName() + ": " + cookie.getValue());
}
}
System.out.println();
System.out.println("[Content 편의 조회]");
System.out.println("request.getContentType() = " + request.getContentType()); //get방식 조회시 content를 거의 안 보내기 때문에 null 나올 것
System.out.println("request.getContentLength() = " + request.getContentLength());
System.out.println("request.getCharacterEncoding() = " + request.getCharacterEncoding());
System.out.println("--- Header 편의 조회 end ---");
System.out.println();
}
}
기타 서버 관련 정보도 확인가능하다.
[RequestHeaderServlet.java]
@WebServlet(name = "requestHeaderServlet", urlPatterns = "/request-header")
public class RequestHeaderServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
printStartLine(request);
printHeaders(request);
printHeaderUtils(request);
printEtc(request);
}
//HTTP - 스타트라인 출력
//...
//HTTP - Header 모든 정보
//...
//Header 편리한 조회
//...
//기타 정보(네트워크 커넥션 등에 대한 정보)
private void printEtc(HttpServletRequest request) {
System.out.println("--- 기타 조회 start ---");
System.out.println("[Remote 정보]");
System.out.println("request.getRemoteHost() = " + request.getRemoteHost()); //
System.out.println("request.getRemoteAddr() = " + request.getRemoteAddr()); //
System.out.println("request.getRemotePort() = " + request.getRemotePort()); //
System.out.println();
System.out.println("[Local 정보]");
System.out.println("request.getLocalName() = " + request.getLocalName()); //
System.out.println("request.getLocalAddr() = " + request.getLocalAddr()); //
System.out.println("request.getLocalPort() = " + request.getLocalPort()); //
System.out.println("--- 기타 조회 end ---");
System.out.println();
}
}
5~8.Http 요청 데이터 - 개요, GET쿼리파라미터, POST HTML Form, API 메시지 바디(단순 text)
클라이언트에서 서버로 데이터 전달 방법은 크게 3가지가 있다.
- 1)GET - 쿼리 파라미터
- /url?username=hello&age=20과 같이 url에 붙여 보내는 방식
- 메세지 바디가 없다(Content-Type도 당연히 없음)
- 검색, 필터, 페이징 등에서 사용
- 요청 데이터 뽑을 때는 request.getParameter() 사용
- 요청데이터 중복일 때 request.getParameterValues() 사용
- 2)POST - HTML Form
- Content-Type : application/x-www-form-urlencoded
- 메세지 바디에 쿼리 파라미터 형식으로 전달 username=hello&age=20
- 회원가입, 상품주문 등
- POST도 메세지 바디에 쿼리파라미터 형식으로 보내기 때문에 request.getParameter() 사용 가능
- 3)HTTP message body에 데이터 직접 담아서 요청
- HTTP API에서 주로 사용 : JSON, XML
- POST, PUT, PATCH
HTTP message body에 직접 담아서 요청을 보낼 때 다음과 같이 꺼내 쓸 수 있다.
[RequestBodyStringServlet.java]
@WebServlet(name = "requestBodyStringServlet", urlPatterns = "/request-body-string")
public class RequestBodyStringServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//메세지 바디를 byte로 얻음(inputStream으로 데이터를 읽을 수 있음)
ServletInputStream inputStream = request.getInputStream();
//스프링 제공 유틸리티
String messageBody = StreamUtils.copyToString(inputStream, StandardCharsets.UTF_8);
System.out.println("messageBody = " + messageBody);
response.getWriter().write("ok");
}
}
9.HTTP 요청 데이터 - API 메시지 바디 - JSON
- Content-type : application/json
- message body : {"username": "hello", "age":20}
JSON 형식 파싱을 위한 객체 추가를 한다.
[HelloData.java]
@Getter
@Setter
public class HelloData {
private String username;
private int age;
}
JSON 요청을 읽기 위한 서블릿을 추가한다.
[RequestBodyJsonServlet.java]
/**
* 제이슨 형식의 메시지 바디를 파싱해본다
*/
@WebServlet(name = "requestBodyJsonServlet", urlPatterns = "/request-body-json")
public class RequestBodyJsonServlet extends HttpServlet {
private ObjectMapper objectMapper = new ObjectMapper();
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
ServletInputStream inputStream = request.getInputStream();
String messageBody = StreamUtils.copyToString(inputStream, StandardCharsets.UTF_8);
System.out.println("messageBody =" + messageBody);
//jackson라이브러리로 vo 파싱
HelloData helloData = objectMapper.readValue(messageBody, HelloData.class);
System.out.println("helloData.username = " + helloData.getUsername());
System.out.println("helloData.age = " + helloData.getAge());
response.getWriter().write("ok");
}
}
json은 jackson라이브러리로 파싱한다. 위에서 ObjectMapper를 사용한다.
10.HttpServletResponse - 기본 사용법
HttpServletResponse는 응답관련된 헤더 셋팅이나 쿠키값,리다이렉트(302) 등을 사용할 수 있다.
[ResponseHeaderServlet.java]
@WebServlet(name = "responseHeaderSerlvet", urlPatterns = "/response-header")
public class ResponseHeaderServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//[status-line]
response.setStatus(HttpServletResponse.SC_OK);
//[response-headers]
response.setHeader("Content-Type", "text/plain;charset-utf-8");
response.setHeader("Cache-Control", "no-cache, no-store, must-revalidate");
response.setHeader("Pragma", "no-cache");
response.setHeader("my-header", "hello"); //커스텀 헤더도 만들 수 있다.
//[Header 편의 메서드]
content(response);
cookie(response);
redirect(response);
//[message body]
PrintWriter writer = response.getWriter();
writer.println("ok");
}
private void content(HttpServletResponse response) {
response.setContentType("text/plain");
response.setCharacterEncoding("utf-8");
//response.setContentLength(2); //생략시 자동 생성
}
private void cookie(HttpServletResponse response) {
//기존 방법
//response.setHeader("Set-Cookie", "myCookie=good; Max-Age=600");
//편리한 방법
Cookie cookie = new Cookie("myCookie", "good");
cookie.setMaxAge(600);
response.addCookie(cookie);
}
private void redirect(HttpServletResponse response) throws IOException {
//Status Code 302
//Location: /basic/hello-form.html
//기존 방법
// response.setStatus(HttpServletResponse.SC_FOUND);
// response.setHeader("Location", "/basic/hello-form.html");
//편리한 방법
response.sendRedirect("/basic/hello-form.html");
}
}
11. HTTP 응답 데이터 - 단순 텍스트, HTML
단순 텍스트 응답과 HTML 응답은 다음과 같이 PrintWriter를 통해 생성하여 소스보기로 처리할 수 있다.
[ResponseHtmlServlet.java]
@WebServlet(name = "responseHtmlServlet", urlPatterns = "/response-html")
public class ResponseHtmlServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//Content-Type : text/html;charset=utf-8
response.setContentType("text/html");
response.setCharacterEncoding("utf-8");
//화면에 안녕?이 뿌려짐. 소스보기하면 HTML구성
PrintWriter writer = response.getWriter();
writer.println("<html>");
writer.println("<body>");
writer.println("<div>안녕?</div>");
writer.println("</body>");
writer.println("</html>");
}
}
12. HTTP 응답 데이터 - API JSON
Content-Type은 "application/json"을 사용한다. 자체적으로 json은 캐릭터셋이 UTF-8이기 때문에 CharacterEncoding을 굳이 붙여줄 필요는 없다.
다음과 같이 HelloData 객체 setter에 값을 넣은 후, jackson 라이브러리의 ObjectMapper를 사용한다.
[ResponseJsonServlet.java]
@WebServlet(name = "responseJsonServlet", urlPatterns = "/response-json")
public class ResponseJsonServlet extends HttpServlet {
private ObjectMapper objectMapper = new ObjectMapper();
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//Content-Type: application/json
//json은 자체적으로 utf-8 캐릭터셋이 기본이라서 뒤에 utf-8붙여주는 것이 의미는 없음
response.setContentType("application/json");
response.setCharacterEncoding("utf-8");
HelloData helloData = new HelloData();
helloData.setUsername("kim");
helloData.setAge(20);
//{"username":"kim", "age":20}
//스프링에서는 return helloData로 json을 반환할 수 있음
String result = objectMapper.writeValueAsString(helloData);
response.getWriter().write(result);
}
}
'개발자 일지 > Spring' 카테고리의 다른 글
[인프런 김영한 로드맵4]스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술(6) (0) | 2022.06.18 |
---|---|
[인프런 김영한 로드맵4]스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술(5) (0) | 2022.06.14 |
[인프런 김영한 로드맵4]스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술(4) (0) | 2022.06.04 |
[인프런 김영한 로드맵4]스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술(3) (2) | 2022.05.31 |
[인프런 김영한 로드맵4]스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술(1) (0) | 2022.05.12 |
[인프런 김영한 로드맵2]스프링 핵심 원리 정리(6) (0) | 2022.03.20 |
[인프런 김영한 로드맵2]스프링 핵심 원리 정리(5) (0) | 2022.03.05 |
[인프런 김영한 로드맵2]스프링 핵심 원리 정리(3) (0) | 2022.02.04 |