H'academy
  • H'academy
  • Web
    • Back-end
      • Spring Framework
        • Spring 개발환경 구축
          • Project 생성
          • Maven 설정
          • Maven 저장소 위치 설정
          • Apache Tomcat 연동
          • Lombok 설정
        • Spring MVC Project
          • pom.xml
          • web.xml
          • root-context.xml
          • servlet-context.xml
          • DispatcherServlet
        • Controller
          • File Upload / Download
          • RestController(작성중)
        • Spring Test
          • Assert 단정문
          • Spring 환경 연동 테스트
          • MockMvc Test
        • Spring AOP
          • AOP 용어
          • Pointcut Expression
          • Filter
          • Interceptor
          • Filter vs Interceptor
          • Controller Advice(작성중)
        • Spring JDBC
          • DBCP
          • myBatis
            • ResultMap 활용
        • ETC
          • Logging
            • Tomcat Logging 설정
          • Property 관리
          • E-mail
            • Gmail 변경사항
          • Scheduler
          • Websocket
      • Spring Boot
        • STS 다운로드
        • 프로젝트 생성
        • 프로젝트 구조
        • Spring Devtools
        • Spring Controller
          • JSP 연동
            • * JSP 생성불가 문제
            • Spring Boot 3.x
          • Model
          • 요청 파라미터 처리
          • 경로 변수 처리
          • 요청 메소드
          • Redirect
          • Spring Actuator
        • Spring Boot JDBC
          • Database 정보
          • DAO Pattern
          • ORM Framework
            • myBatis
              • Mapper 생성
            • Spring Data JPA
        • Spring AOP
          • AspectJ
          • Filter
          • Interceptor
          • ControllerAdvice
        • REST API
        • Spring WebSocket
        • Send Mail
        • Logging
        • Spring Boot Test
        • Lombok
      • Java EE
        • 개발환경 구성
          • Server와 Client
          • Web Server와 WAS
        • Dynamic Web Project
      • Apache Tomcat
        • Apache Tomcat 구조
        • Tomcat User 생성
        • Tomcat Manager 설정
      • Maven
      • Jenkins
        • Installation
          • Github Webhook 설정
          • Maven 연동
        • Project setting
          • 소스 코드 관리 설정
          • 빌드 유발 설정
          • 빌드 작업 설정
      • OpenAPI(swagger)
        • swagger 2.x (legacy)
        • swagger 3.x (boot)
        • Swagger UI 꾸미기
        • SpringDoc
    • Front-end
      • HTML
        • Heading
        • <A>
        • <IMG>
        • <DIV>
        • <TABLE>
        • <FORM>
          • <INPUT>
          • <SELECT>
          • <TEXTAREA>
          • <FIELDSET>
        • List
      • CSS
        • CSS 적용 방식
          • 크기(size)
            • 픽셀(px)
          • 색상(color)
        • CSS 선택자
          • 스타일 우선순위
        • CSS 주요 속성
          • 크기 속성
            • border
            • margin
            • padding
            • box-sizing
          • 배경 속성
          • 글자 속성
            • font
            • text
          • 배치 속성
            • position
            • display
            • overflow
            • float
            • flexbox
        • CSS Reset
      • JS
        • syntax
          • let, const
          • 구조 분해 할당
          • spread operator
          • Object copy
          • this
          • module
        • jQuery
          • structure
          • function
          • ajax
            • ajax file upload
        • VueJS
          • CDN(Vue3)
            • 기본 구조
            • Vue Devtools
            • jQuery vs VueJS
            • Vue instance
              • data
                • IME issue
              • v-model
              • template
              • computed
              • watch
                • deep watch
                • vs Computed
              • methods
            • Vue directive
              • v-text
              • v-html
              • v-bind
                • v-bind style
                • v-bind class
              • v-on
                • once
                • prevent
              • v-show
              • v-if
              • v-for
          • Vue-cli 3
            • 프로젝트 구조
            • src
              • main.js
              • App.vue
              • HelloWorld.vue
            • Vue Router 4
              • History mode
              • 404 not found
              • Guard
                • beforeEach
            • Vuex
              • 이동 횟수 측정
            • axios
              • interceptor
          • Vue-electron
        • ReactJS
          • CDN(React18)
            • 템플릿 설명
            • JSX
            • render
            • Handling Events
              • onClick
              • onInput
            • Component
              • 클래스 컴포넌트
                • state
                • props
                • event
                  • Mouse
                  • Input
                • lifecycle
              • 함수형 컴포넌트
                • useState
                • useEffect
                • useReducer
                • useMemo
                • props
                • 조건부 렌더링
          • CRA
            • 자원 경로 설정
            • src 절대 경로 설정
            • react-router
            • Context API
            • Electron 설정
        • Webpack
          • NodeJS
          • Git
    • Ubuntu server
      • Jenkins
    • 테스트 도구
      • JMeter
        • Test Plan
        • Test Plan(with login)
        • Plugin 설치
    • WebRTC
  • Base Language
    • Java
      • Java 실행 과정
        • 개발 환경 구축
        • 샘플 코드 작성
        • 컴파일/실행
        • 환경 변수 설정
      • Java 초급
        • Hello world
        • 자료와 정보
        • 변수
          • 정수
            • 정수의 저장 원리
            • 정수형 변수 생성
          • 실수
            • 실수의 저장 원리
            • 실수형 변수 생성
          • 논리
            • 논리형 변수 생성
          • 문자
          • 문자열
            • 문자열 변수 생성
          • 사용자 정의 자료형
        • 연산자
          • 대입 연산
          • 부호 연산
          • 산술 연산
          • 복합 대입 연산
          • 증감 연산
          • 쉬프트 연산
          • 비트 연산
          • 비교 연산
          • 논리 연산
          • 삼항 연산
          • 변환 연산
        • 표준 입출력
          • 표준 출력
          • 표준 오류 출력
          • 표준 입력
            • Scanner 입력
        • 제어문
          • if 구문
            • 단일 조건
            • 두 가지의 조건
            • 여러 가지의 조건
          • switch~case 구문
          • while 구문
          • do~while 구문
          • for 구문
        • 랜덤(Random)
        • 배열
          • 1차원 배열
          • 2차원 배열
          • 다차원 배열
          • Sort
            • 선택 정렬
            • 버블 정렬
            • 삽입 정렬
      • Java 중급
        • 객체 지향 프로그래밍
        • 클래스
        • 인스턴스
        • 필드
        • 메소드
          • 설정(setter) 메소드
          • 반환(getter) 메소드
        • 생성자
        • 접근 제한
          • private
          • package
          • protected
          • public
        • 정적(static)
        • 불변(final)
        • 상수(constant)
        • 상속
          • this와 super의 이해
          • 멤버 변수 설정
          • 멤버 메소드 설정
          • 생성자 설정
        • 추상화
          • 추상 클래스
          • 인터페이스
        • 다형성
        • 중첩 클래스
          • 일반 중첩 클래스
          • static 중첩 클래스
          • 지역 중첩 클래스
          • 익명 중첩 클래스
      • Java API(작성중)
        • java.lang
        • java.util
        • java.text
        • java.time
        • java.io
        • java.net
        • java.sql
        • java.awt
        • javax.swing
      • Java 고급
        • 제네릭(Generic)
        • 열거형(Enum)
        • Annotation Type
          • Retention
          • Target
          • Nested Annotation
        • Java Reflection
          • Intro
          • Class
          • Field
          • Constructor
          • Method
          • Package
      • JDBC
        • 드라이버 로드
        • 연결 생성
          • JDBC Driver 유형
        • JdbcTemplate
      • Java 버전별 변경사항
        • Java 1.8
          • Lambda Expression
          • Stream API
          • java.time 패키지
        • Java 9
        • Java 12
          • switch~case
        • Java 13
          • Text Block
    • Arduino
      • 학습 준비
        • Reference
        • 아두이노 온라인 시뮬레이터
        • 전기의 이해
        • 그라운드(GND)의 이해
      • 개발 환경 구축
        • CH340 보드 설치
        • Editor 글꼴 설정
      • 아두이노 제어
        • 코드의 구성
        • 시리얼 출력
        • 디지털 출력
          • LED 출력
        • PWM 출력
        • 시리얼 입력
        • 디지털 입력
          • PIR 센서
        • 아날로그 입력
          • 온도 센서(LM35DZ)
          • 조도 센서
          • 가스 센서
        • 피에조 부저
        • 서보 모터 제어
          • 시리얼 서보 모터 제어
          • 스위치 서보 모터 제어
        • 온습도 센서(DHT-11)
        • LCD
      • 아두이노 통신
        • 블루투스(Bluetooth)
          • HC-06
            • 온도 센서 활용
          • HM-10
        • 와이파이(Wi-Fi)
          • ESP-01
            • WiFiEsp 라이브러리 추가
            • WiFiEsp 네트워크 스캔
        • 이더넷(Ethernet)
        • ArduinoJson 라이브리 사용
    • Regular Expression
  • Database
    • Oracle
      • 설치
        • SQL Plus
        • SQL Developer
        • DBeaver
        • Docker(Mac OS)
      • SQL
        • 관리자 명령
          • 사용자 관리
          • 권한 관리
          • Encoding 설정
          • 서비스 포트 설정
        • 사용자 명령
          • 테이블 관리
            • 테이블 항목 제약 조건
            • 외래키 제약조건
            • 데이터 유형
          • 시퀀스 관리
          • 데이터 관리
            • 데이터 추가
            • 데이터 조회
              • 오라클 내장 함수
              • 그룹 조회
              • Top N Query
            • 데이터 수정
            • 데이터 삭제
          • 뷰 관리
          • 인덱스 관리
          • 집합 연산
          • 테이블 조인
            • 등가 조인
            • 외부 조인
        • HR 계정
        • 기타 명령
          • 컬럼 간격 조정
      • 내보내기 및 불러오기
        • EXPDP, IMPDP
        • Docker로 실행중인 경우
      • 기타
        • 자동 세션 제거
        • Tablespace 관리
        • Unlock user
  • 면접대비
    • Q&A
      • Java
      • Database
      • Web
      • IT 일반상식
  • ETC
    • Eclipse IDE
      • 설치
      • 화면 설명
      • 환경 설정
      • 프로젝트 생성
      • 소스파일 작성 및 실행
    • Github
      • Github 계정 관리
      • Github 저장소 관리
        • Profile
        • Collaborators
        • Code
        • Issues
        • Pull requests
        • Action
          • Vue 3 Deploy Action
      • Personal Access Token
      • Git ignore 설정
        • Spring 개발용 샘플
      • Github Eclipse 연동
        • Branches
        • Project
        • Team project
          • Dynamic web project
          • Spring Boot Project
          • Branch 생성
          • Pull Request
          • Branch 보호
          • Branch 다시 생성
    • Figma
      • Team
      • Design
      • Component
    • 과정평가형
      • 실기 모의고사 1회
      • 실기 모의고사 2회
      • 실기 모의고사 3회
      • 실기 모의고사 4회
    • VScode
      • Live Server 실행 오류
      • Github 계정 변경
    • Naver ToastUI Editor
      • Editor 만들기
        • Editor options
          • File upload
      • Viewer 만들기
    • 문자 인코딩 방식
    • OBS
      • 녹화 잡음 제거 설정
    • Semantic Version
Powered by GitBook
On this page
  • DAO란
  • DB 정보
  • DAO 구조
  • 프로젝트 생성
  • application.properties
  • DTO 생성
  • DAO 생성
  • Controller 생성
  • 클래스 구현체 변수 생성
  • 추상 인터페이스 변수 생성
  • StudentDao 메소드
  • 등록 메소드
  • 수정 메소드
  • 삭제 메소드
  • 조회 메소드
  • 상세 메소드
  1. Web
  2. Back-end
  3. Spring Boot
  4. Spring Boot JDBC

DAO Pattern

이 문서에서는 DAO(Data Access Object)의 필요성에 대해서 설명한다

PreviousDatabase 정보NextORM Framework

Last updated 2 years ago

DAO란

DAO는 Data Access Object의 줄임말로 데이터베이스와 관련된 CRUD 작업을 처리하는 도구를 말한다. DAO를 사용하면 데이터베이스 작업을 기능별로 모듈화 할 수 있다.

DB 정보

Database 정보를 기반으로 진행한다

DAO 구조

Spring은 다음과 같은 특징을 가지고 있다.

  • IoC - 개발자가 사용하고자 하는 객체는 Spring Container에 등록해서 사용해야 한다.

  • DI - 등록된 객체들은 서로 주입하여 사용할 수 있다.

따라서 DAO를 Spring에 등록해야 하며, 사용하고자 하는 위치에 주입해야 한다.

프로젝트 생성

application.properties

JDBC 사용을 위해 application.propertiees 파일에 다음과 같이 설정한다.

# jdbc
spring.datasource.driver-class-name=oracle.jdbc.OracleDriver
spring.datasource.url=jdbc:oracle:thin:@localhost:1521:xe
spring.datasource.username=hacademy
spring.datasource.password=hacademy

# actuator
management.endpoints.web.exposure.include=mappings,beans,env

DTO 생성

student테이블의 데이터를 변환하기 위하여 StudentDto 클래스를 생성한다.

StudentDto.java
package com.hacademy.boot06.entity;

public class StudentDto {
    private int no;
    private String name;
    private int korean, english, math;
    public StudentDto() {}
    //setter, getter, toString()...
}

DAO 생성

DAO는 PSA(Portable Service Abstraction) 구조로 생성한다. 즉, 인터페이스와 클래스로 분리하여 생성한다.

  • StudentDao - 인터페이스로 생성하며 기능의 명세를 정의

  • StudentDaoImpl - 클래스로 생성하며 인터페이스를 상속받아 기능의 실제 구현을 정의

이와 같이 구현하면 구현체를 교체하는 것이 수월해진다. 이 내용에 대해선 잠시 후에 다룬다.

package com.hacademy.boot06.repository;

public interface StudentDao {
    //추상 메소드(기능 명세) 정의
}
package com.hacademy.boot06.repository;

import org.springframework.stereotype.Repository;

@Repository
public class StudentDaoImpl implements StudentDao{
    //기능 명세에 대한 실제 구현 코드
}

StudentDaoImpl의 상단에 @Repository 애노테이션이 작성되어 있으며, 이로 인하여 구현체인 StudentDaoImpl이 Spring Container에 등록된다. StudentDao는 인터페이스므로 등록할 수 없다.

등록된 객체간에 의존성 주입(DI)이 가능하므로 다음과 같이 구현체 클래스에 JdbcTemplate을 사용하도록 선언할 수 있다.

@Autowired
private JdbcTemplate jdbcTemplate;

Controller 생성

Student와 관련된 처리를 수행하는 StudentController를 생성한다.

package com.hacademy.boot06.controller;

import org.springframework.stereotype.Controller;

@Controller
public class StudentController {

}

Database 관련 처리를 위하여 StudentDaoImpl 클래스를 주입하여야 한다. 변수 선언 방법은 두 가지가 있다.

클래스 구현체 변수 생성

@Autowired
private StudentDaoImpl studentDao;

컨트롤러에 위와 같이 변수를 선언할 경우 주입은 정상적으로 이루어지나 다음과 같은 문제가 발생한다.

  • 추후 StudentDaoImpl 클래스를 다른 클래스로 교체할 경우 해당 변수를 수정해야 한다.

웹 서비스의 특성상 운영기간이 길고 이에 따른 환경변화가 있을 수 있기 때문에 해당 클래스를 직접 주입하는 경우 결합도가 강해져서 확장성이 떨어진다.

추상 인터페이스 변수 생성

@Autowired
private StudentDao studentDao;

컨트롤러에 인터페이스 형태로 변수를 선언할 경우 @Autowired 애노테이션이 해당 타입과 상속받은 타입에 대한 스캔을 하여 가능한 객체를 주입한다. 따라서 등록된 StudentDaoImpl 클래스의 객체가 주입된다.

StudentDao 메소드

Database 작업을 처리하는 StudentDao에는 기본적으로 다음과 같은 메소드가 포함된다. 필요하다면 다른 메소드를 더 추가하여 사용할 수 있다.

public interface StudentDao {
    void insert(StudentDto studentDto);//등록(C)
    boolean update(StudentDto studentDto);//수정(U)
    boolean delete(int no);//삭제(D)
    List<StudentDto> selectList();//목록(R)
    StudentDto selectOne(int no);//상세(R)
}

등록 메소드

등록을 위한 SQL 구문은 다음과 같다.

INSERT INTO STUDENT(NO, NAME, KOREAN, ENGLISH, MATH)
VALUES(STUDENT_SEQ.NEXTVAL, ?, ?, ?, ?)

구문에 시퀀스를 제외한 이름, 국어점수, 수학점수, 영어점수를 채워야 하므로 다음과 같이 메소드를 구성할 수 있다.

void insert(String name, int korean, int english, int math);

하지만 name, korean, english, math는 StudentDto의 필드이므로 StudentDto 객체 형태로 선언하면 한 번의 선언으로 데이터를 묶어서 전달받을 수 있다.

void insert(StudentDto studentDto);

처리 코드가 위치하는 StudentDaoImpl 클래스의 구현 코드는 다음과 같다.

void insert(StudentDto studentDto){
    String sql = "INSERT INTO STUDENT(NO, NAME, KOREAN, ENGLISH, MATH) "
                + "VALUES(STUDENT_SEQ.NEXTVAL, ?, ?, ?, ?)";
    Object[] param = {
            studentDto.getName(), studentDto.getKorean(), 
            studentDto.getEnglish(), studentDto.getMath()
    };
    jdbcTemplate.update(sql, param);
}

수정 메소드

UPDATE STUDENT SET NAME=?, KOREAN=?, ENGLISH=?, MATH=? WHERE NO=?

특정 대상을 수정하려면 Primary key인 번호(no)가 필요하다. 따라서 번호를 제외한 다른 정보 중 원하는 것을 수정할 수 있어야 하며, 하나만 만들 예정이므로 번호를 제외한 모든 정보가 변경되도록 처리한다. 데이터는 번호, 이름, 국어점수, 수학점수, 영어점수 총 5개가 필요하다. 번호가 존재하지 않을 경우 실행이 되어도 수정된 데이터가 없을 수 있으므로 반환형을 boolean으로 설정하여 성공/실패 여부를 반환할 수 있도록 구현한다.

객체를 사용하지 않을 경우의 메소드 형태는 다음과 같다.

boolean update(int no, String name, int korean, int english, int math);

객체를 사용할 경우의 메소드 형태는 다음과 같으며, 형태가 간결하여 사용을 권장한다.

boolean update(StudentDto studentDto);

처리 코드가 위치하는 StudentDaoImpl 클래스의 구현 코드는 다음과 같다.

public boolean update(StudentDto studentDto) {
    String sql = "UPDATE STUDENT "
                    +"SET NAME=?, KOREAN=?, ENGLISH=?, MATH=? "
                    +"WHERE NO=?";
    Object[] param = {
        studentDto.getName(), studentDto.getKorean(),
        studentDto.getEnglish(), studentDto.getMath(),
        studentDto.getNo()
    };
    return jdbcTemplate.update(sql, param) > 0;
}

삭제 메소드

특정 대상을 삭제하려면 Primary key가 필요하다.

DELETE STUDENT WHERE NO = ?

Unique 항목으로도 가능하지만 현재 테이블은 Primary key인 번호(no)를 제외한 다른 Unique 항목이 없으므로 Primary key를 사용한다. 번호에 따라 대상이 존재하지 않아 삭제가 이루어지지 않을 수 있으므로 반환형을 boolean으로 설정하고 성공/실패 여부를 반환하도록 설정한다.

메소드의 형태는 다음과 같다.

boolean delete(int no);

처리 코드가 위치하는 StudentDaoImpl 클래스의 구현 코드는 다음과 같다.

public boolean delete(int no) {
    String sql = "DELETE STUDENT WHERE NO = ?";
    Object[] param = {no};
    return jdbcTemplate.update(sql, param) > 0;
}

조회 메소드

SELECT * FROM STUDENT

전체 조회를 하는 경우는 드물지만 구현한다면 위의 SQL 구문과 같다. 목록의 경우 정렬이 반드시 들어가는것이 좋기 때문에 Primary key인 번호(no)를 이용하여 다음과 같이 정렬 후 조회할 수 있다.

SELECT * FROM STUDENT ORDER BY NO ASC

데이터베이스 조회 결과(ResultSet)을 List로 변환하기 위해서는 RowMapper<T> 객체가 필요하다.

private RowMapper<StudentDto> mapper = new RowMapper<StudentDto>() {
    @Override
    public StudentDto mapRow(ResultSet rs, int rowNum) throws SQLException {
        StudentDto studentDto = new StudentDto();
        studentDto.setNo(rs.getInt("no"));
        studentDto.setName(rs.getString("name"));
        studentDto.setKorean(rs.getInt("korean"));
        studentDto.setEnglish(rs.getInt("english"));
        studentDto.setMath(rs.getInt("math"));
        return studentDto;
    }
};

메소드는 List 타입이 반환되도록 다음과 같이 구성한다. 홀더가 없으므로 매개변수는 작성하지 않는다.

List<StudentDto> selectList();

처리 코드가 위치하는 StudentDaoImpl 클래스의 구현 코드는 다음과 같다.

@Override
public List<StudentDto> selectList() {
    String sql = "SELECT * FROM STUDENT ORDER BY NO ASC";
    return jdbcTemplate.query(sql, mapper);
}

상세 메소드

상세 조회는 Primary key를 이용하여 단 하나의 데이터를 조회한다.

SELECT * FROM STUDENT WHERE NO = ?

따라서 결과는 없거나 혹은 한 개가 나오며, 이에 대한 자료형은 StudentDto이다. 따라서 메소드의 형태는 다음과 같다.

StudentDto selectOne(int no);

그리고 RowMapper처럼 ResultSet의 데이터 유무에 따라 StudentDto로 변환해줄 ResultSetExtractor<StudentDto> 객체가 필요하다.

private ResultSetExtractor<StudentDto> extractor = new ResultSetExtractor<StudentDto>() {
    @Override
    public StudentDto extractData(ResultSet rs) throws SQLException, DataAccessException {
        if(rs.next()) {
            StudentDto studentDto = new StudentDto();
            studentDto.setNo(rs.getInt("no"));
            studentDto.setName(rs.getString("name"));
            studentDto.setKorean(rs.getInt("korean"));
            studentDto.setEnglish(rs.getInt("english"));
            studentDto.setMath(rs.getInt("math"));
            return studentDto;
        }
        else {
            return null;
        }
    }
};

처리 코드가 위치하는 StudentDaoImpl 클래스의 구현 코드는 다음과 같다.

public StudentDto selectOne(int no) {
    String sql = "SELECT * FROM STUDENT WHERE NO = ?";
    Object[] param = {no};
    return jdbcTemplate.query(sql, extractor, param);
}