JPA로 DB 연결
- bootbasic 프로젝트를 생성한다.
data:image/s3,"s3://crabby-images/ee6e0/ee6e0a3f5bab501eb26be6aadb860b8ab0e81d70" alt=""
data:image/s3,"s3://crabby-images/fd19e/fd19eff0b87f45568b790530b3f840edea24701e" alt=""
-
Spring Boot 의 기본 버전 설정과 DB와 연결하기 위한 Spring Data JPA, MySQL Driver, Developer Tools > Lombok에 체크한다.
-
Spring Boot Version은 2.6.10이고 Spring Boot에서 Web을 사용하기 위해 Spring Boot DevTools와 Spring Web에 체크를 해주어야한다.
data:image/s3,"s3://crabby-images/12c91/12c91eb7c9ccd28307b936712526303dccc67d31" alt=""
- JPA로 DB에 연결하기 위해 체크해준다.
data:image/s3,"s3://crabby-images/cd815/cd815785823291d1194886e8bd9518d77558e620" alt=""
** JPA 모듈을 사용하게되면 Connection Pool과 DB 연결을 자동으로 해주는데, 다만 프로젝트 실행 시 DB 연결이 반드시 선행작업이 되어야한다.
- DB는 MySQL을 사용하기때문에 자바와 MySQL을 연결해주는 Driver에 체크한다.
data:image/s3,"s3://crabby-images/088bb/088bbabbfee6a74dd98275d61e171a03bdaef05b" alt=""
- Vo를 만들 때 @(어노테이션)으로 쉽게 Getter, Setter, ToString 등을 만들기 위해 체크해준다.
data:image/s3,"s3://crabby-images/c8fa7/c8fa71c19ae9a02a74e125ebcf4545c35fdfc1f8" alt=""
- Selected된 것을 확인하고 Finish 버튼을 클릭한다.
data:image/s3,"s3://crabby-images/5f967/5f9677257bd799b3d8c08b70904d4a301b5e64eb" alt=""
- application.properties 에 DB에 연결하기위한 driver, url, username, password를 입력해주어야한다.
data:image/s3,"s3://crabby-images/4b73d/4b73d51bc097bcd8bf43b6a41bc3f2a3dad79667" alt=""
- MySQL의 Workbench를 이용하여 DB생성과 계정을 생성하고 모든 권한을 부여해준다.
data:image/s3,"s3://crabby-images/b923b/b923b17b9fa0c2fc9d517d4065cf7e64c2ab09b3" alt=""
- 만든 계정으로 Workbench에 Connection Card를 추가한다.
data:image/s3,"s3://crabby-images/12286/122865041e0ba884f7306ad25b554fecc996f19c" alt=""
data:image/s3,"s3://crabby-images/8348d/8348d2688d410c0be6460859cec28aa6821e7ea1" alt=""
- application.properties 에 username과 password를 수정해준다.
data:image/s3,"s3://crabby-images/192b2/192b296f6897fa394079236a1e96e770cebd8136" alt=""
** 여기까지해주면 JPA 모듈이 데이터베이스와 자동으로 연결해준다.
- DB에 연결이 되었는지 확인해보자.
data:image/s3,"s3://crabby-images/6dbf7/6dbf7332e90aac80051651b7ae4c31acc5377763" alt=""
data:image/s3,"s3://crabby-images/abea2/abea28e91dc3121f222ac0971cfe9795f23b1462" alt=""
JSP vs Thymeleaf (JSP)
data:image/s3,"s3://crabby-images/56f50/56f50daed44466728a7381d65cba3742c88be65c" alt=""
-
spring-boot-starter-web 에 포함된 tomcat 은 JSP 엔진을 포함하고 있지 않다. jsp 파일은 Springboot 의 templates 폴더안에서 작동하지 않는다. 그래서 jsp를 적용하기 위해서는 아래와 같은 의존성을 추가해야한다.
-
https://mvnrepository.com/ 로 이동해서 검색창에 embed를 입력한다.(Tomcat Embed Jasper를 클릭한다.)
data:image/s3,"s3://crabby-images/c87e9/c87e9d50c51f154048f5c4a5af48721cca76c4bd" alt=""
- Spring Boot의 내부 Tomcat과 버전을 맞추기위해 버전을 확인한다.
data:image/s3,"s3://crabby-images/04958/049582349020e67fb5592fea673a7caf9c783efe" alt=""
- 해당 버전의 Gradle 을 복사해서 build.gradle의 dependencies에 추가해준다.
data:image/s3,"s3://crabby-images/ecef9/ecef93b6729030b30e8c05fca2ae6a2092603083" alt=""
data:image/s3,"s3://crabby-images/3e243/3e2437a0abbb39f94493dc9d051361eca63bdc9f" alt=""
- 마찬가지로 jstl을 사용하기위해 build.gradle의 dependencies에 추가해준다.
data:image/s3,"s3://crabby-images/dba45/dba45abfc41a33dcd644a6d2497f9e77bbc62290" alt=""
data:image/s3,"s3://crabby-images/6b7a4/6b7a46f988a26c9e283dfaa20061c43f14bb8a39" alt=""
data:image/s3,"s3://crabby-images/a04bc/a04bcdc9e31955b8e3270dc73793a788a7d08326" alt=""
** 라이브러리를 추가할 때 version을 적어주지않으면 Spring Boot가 알아서 버전을 맞춰서 다운로드해준다.
data:image/s3,"s3://crabby-images/f1a1f/f1a1ff659efafa1275f65bf03dbda42514c6b3a7" alt=""
- 필요한 라이브러리를 추가하게되면 Gradle을 Refresh 해주어야한다.
data:image/s3,"s3://crabby-images/5d3f0/5d3f0feb3a2d98f0aa3ba7d6c5da298539afaf18" alt=""
** jar 파일이 잘 받아졌는지는 확인하려면 Explorer 퍼스펙티브에 Project and External Dependencies에서 꿀단지 모양의 아이콘이 하얗게 들어왔다면 잘 받아진것이다.
data:image/s3,"s3://crabby-images/b1f5f/b1f5fe037d3ec87e2cd84b82874ad61de21343e9" alt=""
data:image/s3,"s3://crabby-images/7099a/7099ad96856ba6f19caad122a31b520c40c194a0" alt=""
data:image/s3,"s3://crabby-images/c98b4/c98b40d8a13be772bb296d75653548f35157c44a" alt=""
- Spring에서 servlet-context.xml에 InternalResourceViewResolver 는 컨트롤러가 지정한 뷰 이름으로 부터 실제로 사용될 뷰를 선택.
- 컨트롤러가 지정한 뷰 이름 앞뒤로 prefix 프로퍼티와 suffix 프로퍼티를 붙인 값이 실제로 사용될 자원의 경로가 됨.
data:image/s3,"s3://crabby-images/9fb05/9fb053f666a9dbcec3bfd4b4294a34367c559289" alt=""
- 위의 그림과 같이 Spring에서 ViewResolver가 하는 역할을 SpringBoot 에서도 해주어야한다.(application.properties 파일)
data:image/s3,"s3://crabby-images/34ab2/34ab2cb6218973e9098385c81a7bbcfd2df245e9" alt=""
- SpringBoot에서 View를 강제적으로 만들어보자.
- SpringBoot의 초기 프로젝트는 src 폴더아래에 main 폴더밖에 없기때문에 main 폴더아래에 webapp/WEB-INF/views/home.jsp 파일까지 만들어준다.
data:image/s3,"s3://crabby-images/04771/04771bc503637d0be7b4580134a799177213d12f" alt=""
data:image/s3,"s3://crabby-images/00301/00301cf9e54b7612d0ecab2c3bcab4df4d473ef1" alt=""
data:image/s3,"s3://crabby-images/40568/40568f2529c5ceff7799d4d84454a8e796bb41ef" alt=""
- ViewResolver 에게 "home"을 리턴해줄 HomeController를 만든다.
data:image/s3,"s3://crabby-images/198b5/198b5a7ee60fd9221f346a7ff109220f979a2258" alt=""
data:image/s3,"s3://crabby-images/939bf/939bffcc5a5c9aab451441075329572600416e47" alt=""
data:image/s3,"s3://crabby-images/b437a/b437acc88f903b7cd616d8917bd3d9a5178a6525" alt=""
- jsp page가 잘 뜨는지 확인해보자.
data:image/s3,"s3://crabby-images/8c2cd/8c2cd355a8f128da6b5b9a4494535480028062a1" alt=""
data:image/s3,"s3://crabby-images/62fc0/62fc0cf66f4ad74846f8bac734dbb7f44acefd1c" alt=""
여기까지가 Spring Boot에서 jsp 를 view 사용하는 방법이었다.
JSP vs Thymeleaf (Thymeleaf)
- JSP에서 사용했던 ViewResolver 주석처리한다.
data:image/s3,"s3://crabby-images/06eb3/06eb33bec664c00ae79535c6abe3c14cb6580d2c" alt=""
- src/main 폴더 아래에 만들어두었던 webapp/WEB-INF/views/home.jsp 모두를 삭제한다.
data:image/s3,"s3://crabby-images/23567/23567feac48ebdc18dcbf0f3bd04dce41591c55f" alt=""
- build.gradle 에 추가했던 jar 파일들을 주석처리한다.
data:image/s3,"s3://crabby-images/80702/8070273a293cec29363a698f1ead5ea6b4ee55ce" alt=""
- 이제 다시 중앙레파지토리에 가서 입력창에 thymeleaf를 입력한다.
- Spring Boot Starter Thymeleaf를 클릭한다.
data:image/s3,"s3://crabby-images/a063c/a063c2518575b77c4da019e339e28ec55d2230f6" alt=""
- 버전은 지울꺼니까 버전은 아무거나 클릭한다.
- gradle(short) 로 복사해온다.
data:image/s3,"s3://crabby-images/fa64b/fa64b2fd59aef9198b2a3e8cd928023e3501d742" alt=""
- 타임리프 모듈을 build.gradle의 dependencies 안에 붙여넣기한다.
data:image/s3,"s3://crabby-images/2f746/2f746c1a396d0d2ba32b89195e27a50c47aa41ba" alt=""
- gradle을 refresh 해준다.
data:image/s3,"s3://crabby-images/fb106/fb10654b32d89f972655a0d531b871ff320b2478" alt=""
- JSP에서와 같이 jar 파일이 제대로 받아졌나 확인한다.
data:image/s3,"s3://crabby-images/44c9c/44c9c29bc3ea1c750eef8358afcd42a285664e51" alt=""
- thymeleaf는 html 기반이기때문에 html 파일을 생성한다.
- html 파일 이름은 home.html 이다.
data:image/s3,"s3://crabby-images/623de/623de4eabf52db20df146d93ed282ff32e7c6082" alt=""
data:image/s3,"s3://crabby-images/a2fe3/a2fe34849c8411e9da9faff6173bf49e88a8ec2d" alt=""
data:image/s3,"s3://crabby-images/4991d/4991dca6ed5453fa5416ba7270635f981dace124" alt=""
- Controller는 JSP 를 view로 했을 때 만들어두었던 HomeController를 그대로 사용한다.
data:image/s3,"s3://crabby-images/9577e/9577e3efe16176ab7739803644fdc4f7d4cc06bb" alt=""
** 혹시나 페이지를 찾을 수 없다는 404 에러가 난다면 내부 Tomcat Server를 Stop and Start를 하거나 Restart를 해주고 브라우저에서 home.html로 이동해보자.
data:image/s3,"s3://crabby-images/3e567/3e5676996deca5a7f9b75e4e4f25cca180701b58" alt=""
** Spring Boot Project 에서 타임리프 문법은 다음 게시글에서 올리겠다.
IOC Container
- 스프링에서 가장 중요한 컨테이너이다.
- 스프링 프로젝트에서 필요한 자바의 객체를 모두 IOC 컨테이너에 만들어놓는데, 여기서 객체를 우리는 bean이라 부른다.
(즉, 스프링의 동작방식은 IOC 컨테이너에 bean을 미리 생성하고 필요할 때 꺼내서 사용하는 방식이다.)
- IOC : Inversion of Control의 줄임말로 필요한 시점에 특정객체를 이용해서 자바단에 꺼내서 사용할 수 있고, bean 끼리는 DI(의존성 주입) 관계이다. 그래서 @Controller, @Service, @Repository가 모두 DI 관계이다.
- @Controller에서 Autowired로 @Service를 연결하고 @Service는 Autowired로 @Repository(Dao)와 연결하고 Dao는 DB와 연결한다.(의존성관계)
- 스프링에서는 servlet-context.xml 에서 bean 을 만들었었는데, 이를 자바의 객체라 부른다.
- 스프링부트에서는 xml 파일이 .java 파일로 바뀌고, 객체를 자바파일로 꺼내서 사용할 수 있도록 바뀌게 된다.(즉, 가독성이 훨씬 더 좋아진다.)
- 자바 파일을 생성하고 @(어노테이션)만 붙여주면 자동으로 빌드를 생성해준다.
- WebMvcConfigurer는 스프링부트에 내장되어있는 인터페이스이다.
- 클래스가 WebMvcConfigurer를 상속받아서 클래스가 역할을 할 수 있도록 만들어주면 된다.
- WebMvcConfigurer를 상속받은 클래스 안에서 @Bean(빈으로 등록)으로 사용하겠다고 선언을 해준다. 그러면 이 해당 파일을 읽어서 @Bean에 해당하는 것을 객체로 생성해준다.
data:image/s3,"s3://crabby-images/dea1e/dea1e5d69fb9c3794c51bb684ebfc745d5af5eed" alt=""
data:image/s3,"s3://crabby-images/7e003/7e003b08a4f3db3dbc70d23b1737c2fefb495ed1" alt=""
bean을 생성해보자.
- com.simple.basic.config 패키지를 생성한다.
- WebConfig.java 파일을 생성한다.
data:image/s3,"s3://crabby-images/300cb/300cb188a288ad13d412804d2a9334ee66f159c4" alt=""
- 클래스를 설정파일로 선언하기위해 클래스코드 위에 @Configuration 어노테이션을 붙인다.
- WebConfig 클래스는 WebMvcConfigurer 인터페이스를 상속받는다.
data:image/s3,"s3://crabby-images/c2d91/c2d916c039f83f16553413c0e58d86e9bdfc8b5c" alt=""
** WebMvcConfigurer 인터페이스는 하나 예를들자면 스프링에서 servlet-context.xml 파일에서 bean 객체로 InternalResourceViewResolver 클래스가 브라우저의 url을 조각나져있는 경로들을 prefix(접두사)는 WEB-INF/viwes/ 두고, suffix(접미사)는 .jsp로 지정하여 Controller에서 리턴한 값을 실제 jsp의 파일이름과 동일하게 받아서 해당하는 jsp파일로 이동하게끔 해준다. WebMvcConfigurer 인터페이스의 기능은 엄청 많지만 하나의 예를들은 것이다.(필자는 스프링에서의 기본으로 설정한 기능들을 스프링부트에서도 사용할 수 있도록 일반클래스에 WebMvcConfigurer 인터페이스를 상속받음으로써 WebConfig 클래스에서 오버라이딩해서 WebMvcConfigurer의 기능들을 살짝씩 바꿔서 객체를 사용할 수 있다는것까지 이해하고 넘어간다.)
data:image/s3,"s3://crabby-images/18099/180993ebd8ebaf3db4ce8419acfe56a99225cbe2" alt=""
- 잘 동작되는지 내장되어있는 tomcat 서버를 재시작해서 bean 생성 문구가 출력되는지 확인한다.
data:image/s3,"s3://crabby-images/a4e5f/a4e5fd9356b909ad1cf3fa64bf302a6c7385b79d" alt=""
Application Context의 참조
data:image/s3,"s3://crabby-images/66ee0/66ee0884dd9afcf07da239db55735be3fc26508c" alt=""
-
ApplicationContext 객체는 자바로 정의된 것을 뽑을 수 있는 스프링부트에 IOC Container 안에 들어있는 값들을 참조할 수 있는 핵심적인 객체이다.(즉, IOC Comtainer 안에 들어있는 Bean 정보를 실제로 확인할 수 있다.)
-
WebConfig 클래스에서 ApllicationContext 객체를 사용해보자.
-
@Autowired는 필요한 의존 객체의 "타입"에 해당하는 Bean을 찾아 주입한다.(생성자, Setter, 필드)
그렇기때문에 ApplicationContext ctx = new ApplicationContext() 라고 생성하지 않아도 된다.
data:image/s3,"s3://crabby-images/c66be/c66bed9e6f6d73e53dfd0350f024aa8936631bd4" alt=""
-
test 메서드 안에서 IOC Container 안에 들어있는 Bean 들의 숫자를 찍어보자.
data:image/s3,"s3://crabby-images/f994b/f994be5ad992d299b3d4fe4f48a282ae2325ca1d" alt=""
- 어떠한 Bean이 들어있는지는 모르나 230개의 Bean이 들어있다는 것을 확인할 수 있다.
data:image/s3,"s3://crabby-images/27088/2708865306c2b8d5cc8105809f0e930a0f3df4be" alt=""
- HomeController 도 들어있는데, 진짜로 들어있는지 확인해보자.
- Bean의 홈컨트롤러 클래스는 HomeController 타입으로 반환이 된다.
data:image/s3,"s3://crabby-images/13ff6/13ff67f160203c4e26156b261f76a53d76a60732" alt=""
- 이렇게 생성이 되어있는 것을 확인할 수 있다.
- 여기서 중요한것은 new라는 키워드가 없어졌다는 것이다.(스프링에서는 new 대신에 .getBean(xx.class)로 해당 클래스의 Bean을 뽑아올 수 있다.)
data:image/s3,"s3://crabby-images/1c242/1c242c79ee6ce385d720f1f1ab20448f25fd1019" alt=""
- 홈컨트롤러 안에 들어있는 메서드도 확인이 가능하다.
data:image/s3,"s3://crabby-images/994fd/994fde3c85c926fe2923cf9a07e48fcbaae4f8ac" alt=""
data:image/s3,"s3://crabby-images/b95bf/b95bff8ce3fe66d2bb72938de5ddb863bd790678" alt=""
data:image/s3,"s3://crabby-images/f6bea/f6beab6a3d37c40d6c611f040e9dd404b765ccef" alt=""
** 앞으로 사용할 메서드들은 IOC컨테이너에 Bean으로 다 만들어질 것이라는 것을 기억하자.
UtilComponent
- com.simple.basic.util이라는 패키지를 생성한다.
data:image/s3,"s3://crabby-images/42e9c/42e9ca85c24befbda1dc4571dcd9dcbb89154480" alt=""
- 여기다가 UtilComponent라는 클래스를 생성한다.
data:image/s3,"s3://crabby-images/2b20b/2b20b877d6acdc88b4d79a7fa11d6a9da5b8992f" alt=""
- 유틸컴포넌트에 Bean으로 생성해주라는 어노테이션이 있는데 스프링에서 servlet-context.xml 파일에 component-scan이 읽게되는 어노테이션은 1.@Controller, 2.@Service, 3.@Repository, 4.@Component가 있다.
data:image/s3,"s3://crabby-images/58b9b/58b9ba424ff75b42107c291efd1bd2ff4d137aa7" alt=""
- 그래서 직접적으로 @Controller,@Service,@Repository가 아닌 클래스를 Bean으로 만들어주기위해서는 @Component라고 클래스 위에 적어준다.
data:image/s3,"s3://crabby-images/65a07/65a076034a22d2243eef24bc3b5a5977f5b39d53" alt=""
- 여기다가 개별적인 또 다른 기능을 만들수도 있다. 예를들어서 util() 메서드를 만든다.
data:image/s3,"s3://crabby-images/a8f14/a8f1464b22841edf4d0f5b606405e3dfa2f9a161" alt=""
- 빈이 230개에서 231개로 1개 더 늘어난 것을 확인할 수 있다.
data:image/s3,"s3://crabby-images/aa21b/aa21b42cb11d942ddcf20ffe2de7f2d3430d9d1a" alt=""
- Util 메서드를 WebConfig 클래스에서 실행해보자.
data:image/s3,"s3://crabby-images/1ffb4/1ffb4d87bf5a4e46a7a6984d63b7de33e23c602b" alt=""
data:image/s3,"s3://crabby-images/e01c5/e01c51e4e1eca5e1a7af5f0d5d79e87dba576267" alt=""
- 어노테이션으로 Bean을 만들어내는 방법이 아닌 고전적인 방법으로 직접생성해내는 방법도 있다.
- UtilComponent 클래스의 @Component를 주석처리해보자. 당연히 UtilComponent 클래스가 없다고 에러가 날것이다.(그렇게되면 Bean으로 만들어지지않았기때문에 IOC Container 안에 존재하지않기때문이다.)
data:image/s3,"s3://crabby-images/82e70/82e705fc1178d365dbc8264f616ed5224035350e" alt=""
data:image/s3,"s3://crabby-images/53267/5326709a8ed4dfdad9e814b92fcb97ff4a5524ea" alt=""
data:image/s3,"s3://crabby-images/51e9a/51e9aa56bd88c0e8a99781a2886902ed094438cd" alt=""
- WebConfig 클래스에 test()안에 만들었던 UtilComponent 안에 메서드를 확인했던 코드도 주석처리하자.
data:image/s3,"s3://crabby-images/90618/906189ef0e10c6fb2c4a451a83d632aa5cec71e6" alt=""
- 두번째 방법은 WebConfig 클래스 안에서 직접 생성한다.
- 반환유형을 UtilComponent로 하고 utilComponent()라는 메서드를 만들자.
- 리턴에 new UtilComponent()를 써서 객체를 직접생성하고 반환해주는것이다.
data:image/s3,"s3://crabby-images/a00ff/a00ff95b1fcfc766debb650047d11d8667a82aa3" alt=""
- 스프링에서 보았었던 14번 그림을 스프링부트에서는 13번의 그림처럼 바뀌어 들어가는것이다.
data:image/s3,"s3://crabby-images/b96c0/b96c089384cfa1f908007849c8c0e90cc75d8938" alt=""
- 그렇기때문에 아래그림과 같이 스프링부트에서는 빈으로 생성하고 바로 util 메서드를 호출해서 사용할 수 있게되는 것이다.
data:image/s3,"s3://crabby-images/19f9f/19f9fb3482ddb66987debeedb1eda77eb986a08f" alt=""
data:image/s3,"s3://crabby-images/a6c2f/a6c2f307544e0c5f818c3d2068dd1312775e2b28" alt=""
@Value 어노테이션으로 application.properties에 값을 참조하는 방법
- WebConfig 클래스에 url 변수를 만든다.
data:image/s3,"s3://crabby-images/e2f20/e2f200eff15bf86e485701e704e2bf9c9a305066" alt=""
- application.properties에서 아무경로나 적어보자.
data:image/s3,"s3://crabby-images/abe20/abe2092629790512d82a69e4f47353071e8d265a" alt=""
- Value라는 어노테이션으로 application.properties에 있는 값을 참조할 수 있다.
data:image/s3,"s3://crabby-images/c255e/c255ec788d50c56a2b7b159b31b8235de3e5113e" alt=""
- 밑에 있는 test()메서드 Bean에서 값을 찍어보자.
data:image/s3,"s3://crabby-images/89ef6/89ef64aea334aeae5a3a710083ae2db2b6dd9875" alt=""