• 티스토리 홈
  • 프로필사진
    tjdudtn
  • 방명록
  • 공지사항
  • 태그
  • 블로그 관리
  • 글 작성
tjdudtn
  • 프로필사진
    tjdudtn
    • 분류 전체보기 (299)
      • JAVA (155)
        • JAVA 수업 메모 (0)
        • JAVA 수업 파일 (0)
        • JAVA 기초문법 (18)
        • JAVA 객체지향 핵심 (25)
        • JAVA 유용한 클래스 (33)
        • JAVA Swing (18)
      • Data Structure( 자료구조 ) (17)
        • 컬렉션 프레임워크 (4)
        • JDBC (8)
      • 기초 개념 및 환경 설정 (17)
        • HTTP 기초 지식 (사전기반지식) (13)
        • 웹 서버와 웹 애플리케이션 서버(WAS)란? (1)
        • WAS와 웹 컨테이너의 역할 (1)
        • 아파치 톰캣 설치하기 (1)
        • 아파치 톰캣 시작과 폴더 구조 확인 (1)
      • 서블릿 기본 개념과 활용 (8)
        • 서블릿이란? (1)
        • 서블릿 Life Cycle 이란? (1)
        • 서블릿 작성, 배포, web.xml 설정의 이해 (1)
        • Get, Post 요청 방식의 이해 (1)
        • 서블릿과 서블릿 컨텍스트란? (1)
        • 서블릿과 데이터베이스 연동 (1)
        • 잠깐! server.xml과 context.xml.. (1)
        • 서블릿 필터와 리스너란 뭘까? (1)
      • JSP 기본 문법과 구조 (9)
        • JSP(Java Server Pages) 란? (1)
        • JSP 라이프사이클 (1)
        • JSP 기초문법 (1)
        • JSP 주석과 지시자 (1)
        • JSP 기본 태그 ( 스크립트릿, 선언, 표현식 .. (1)
        • JSP 지시자 ( Directive ) 간단 정리 (1)
        • JSP 내장 객체란 뭘까? (1)
        • 폼 처리와 요청 방식 (1)
        • 쿠키와 세션 관리 (1)
      • 서블릿과 JSP의 연동 (7)
        • 서블릿과 JSP의 개념과 차이점 (1)
        • 간단한 게시판 만들어 보기 (1)
        • JSP와 MVC 패턴 Todo 프로젝트 (1)
        • 커스텀 태그 ( JSTL ) 라이브러리 사용, (.. (1)
        • JSTL을 활용한 게시판 기능 만들기 (1)
        • JSP 파일 업로드 (1)
        • JSP 프로그래밍 활용 (1)
      • JavaScript (24)
        • JavaScript 게시판 만들기 (9)
      • Spring Boot (34)
        • Bank App 만들기 (deployment) (31)
      • Flutter (8)
      • MySQL (20)
      • 기술 면접 준비 자료 (0)
  • 방문자 수
    • 전체:
    • 오늘:
    • 어제:
  • 최근 댓글
      등록된 댓글이 없습니다.
    • 최근 공지
        등록된 공지가 없습니다.
      # Home
      # 공지사항
      #
      # 태그
      # 검색결과
      # 방명록
      • JDBC를 활용한 CRUD 와 SOLID 원칙
        2024년 09월 16일
        • tjdudtn
        • 작성자
        • 2024.09.16.:11

        콘솔을 활용한 간단한 퀴즈 게임 만들어 보기

        1. DB , 테이블 설계
        2. 기본 데이터 입력(정규화)
        3. 자바측 라이브러리 설정
        4. 자바측 기능 구현 및 테스트
        5. 리팩토링
        -- 데이터베이스 생성 
        create database quizdb; 
        use quizdb;
        
        drop table quiz;
        
        create table quiz(
        	id int auto_increment primary key, 
        	question varchar(500) not null, 
            answer varchar(500) not null
        );
        
        desc quiz;
        
        -- 기본 샘플 데이터 입력 
        insert into quiz(question, answer ) 
        			values ('대한민국의 수도는?', '서울'),
        					   ('한반도의 남쪽에 위치한 나라는?', '대한민국'),
                    ('세계에서 가능 높은 산은', '에베레스트'),
                    ('태양계의 세 번째 행성은?', '지구'),
                    ('한국의 전통 명절 중 하나로, 음력 8월15일에 해당하는 날은?', '추석'),
                    ('임진왜란 종전 년도는?' ,'159기8'),
                    ('고기압과 저기압에서 바람이 부는 방향은?' ,'고기압');
                            
        select * from quiz;                  
        
        

        프로젝트 구축 및 라이브러리 설정

         

        자바측 기본 코드 입력

        package com.tenco.quiz;
        
        import java.sql.Connection;
        import java.sql.DriverManager;
        import java.util.Scanner;
        
        public class QuizGame {
        	
        	// 준비물 
        	private static final String URL = "jdbc:mysql://localhost:3306/quizdb?serverTimezone=Asia/Seoul";
        	private static final String USER = "root";
        	private static final String PASSWORD = "asd123";
        	
        	public static void main(String[] args) {
        		
        		// JDBC 드리아버 로드 <-- 인터페이스 <--- 구현 클래스 필요 
        		try {
        			Class.forName("com.mysql.cj.jdbc.Driver");
        		} catch (ClassNotFoundException e) {
        			e.printStackTrace();
        			return;
        		}
        		
        		try (Connection conn = DriverManager.getConnection(URL, USER, PASSWORD);
        			  Scanner scanner = new Scanner(System.in)) {
        			
        			while(true) {
        				System.out.println();
        				System.out.println("-----------------------");
        				System.out.println("1. 퀴즈 문제 추가");
        				System.out.println("2. 퀴즈 문제 조회");
        				System.out.println("3. 퀴즈 게임 시작");
        				System.out.println("4. 종료");
        				System.out.print("옵션을 선택 하세요: ");
        				
        				int choice = scanner.nextInt(); // 블로킹 
        				
        				if(choice == 1) {
        					// 퀴즈 문제 추가 --> 함수로 만들기 
        				} else if(choice == 2) {
        					// 퀴즈 문제 조회 --> 함수로 만들기
        				} else if(choice == 3) {
        					// 퀴즈 게임 시작 --> 함수로 만들기
        				} else if(choice == 4) {
        					System.out.println("프로그램을 종료 합니다");
        					break;
        				} else  {
        					System.out.println("잘못된 선택입니다. 다시 시도하세요.");
        				}
        			}
        			
        		} catch (Exception e) {
        			
        		}
        		
        		
        
        	} // end of main 
        
        }
        
        

        1단계 코드 구현 완료

        package com.tenco.quiz;
        
        import java.sql.Connection;
        import java.sql.DriverManager;
        import java.sql.PreparedStatement;
        import java.sql.ResultSet;
        import java.sql.SQLException;
        import java.util.Scanner;
        
        public class QuizGame {
        	
        	// 준비물 
        	private static final String URL = "jdbc:mysql://localhost:3306/quizdb?serverTimezone=Asia/Seoul";
        	private static final String USER = "root";
        	private static final String PASSWORD = "asd123";
        	
        	public static void main(String[] args) {
        		
        		// JDBC 드리아버 로드 <-- 인터페이스 <--- 구현 클래스 필요 
        		try {
        			Class.forName("com.mysql.cj.jdbc.Driver");
        		} catch (ClassNotFoundException e) {
        			e.printStackTrace();
        			return;
        		}
        		
        		try (Connection conn = DriverManager.getConnection(URL, USER, PASSWORD);
        			  Scanner scanner = new Scanner(System.in)) {
        			
        			while(true) {
        				System.out.println();
        				System.out.println("----------------------------------------");
        				System.out.println("1. 퀴즈 문제 추가");
        				System.out.println("2. 퀴즈 문제 조회");
        				System.out.println("3. 퀴즈 게임 시작");
        				System.out.println("4. 종료");
        				System.out.print("옵션을 선택 하세요: ");
        				
        				int choice = scanner.nextInt(); // 블로킹 
        				
        				if(choice == 1) {
        					addQuizQuestion(conn, scanner);
        				} else if(choice == 2) {
        					viewQuizQuestion(conn);
        				} else if(choice == 3) {
        					playQuizGame(conn, scanner);
        				} else if(choice == 4) {
        					System.out.println("프로그램을 종료 합니다");
        					break;
        				} else  {
        					System.out.println("잘못된 선택입니다. 다시 시도하세요.");
        				}
        			}
        			
        		} catch (Exception e) {
        			
        		}
        
        	} // end of main
        
        	private static void playQuizGame(Connection conn, Scanner scanner) {
        		String sql = " select * from quiz order by rand() limit 1  ";
        		
        		try (PreparedStatement pstmt = conn.prepareStatement(sql)) {
        			ResultSet rs = pstmt.executeQuery();
        			
        			if(rs.next()) {
        				
        				String question = rs.getString("question");
        				String answer = rs.getString("answer");
        				
        				System.out.println("퀴즈 문제 : " + question);
        				// 버그처리 
        				scanner.nextLine();
        				System.out.print("당신에 답: ");
        				String userAnswer = scanner.nextLine();
        				
        				if(userAnswer.equalsIgnoreCase(answer)) {
        					System.out.println("정답입니다! 점수를 얻었습니다.");
        				} else {
        					System.out.println("오답입니다!");
        					System.out.println("퀴즈 정답은 -->  " + answer);
        				}
        			} else {
        				System.out.println("죄송합니다 아직 퀴즈 문제를 만들고 있습니다.");
        			}
        		} catch (SQLException e) {
        			e.printStackTrace();
        		}
        	}
        
        	private static void viewQuizQuestion(Connection conn) {
        		String sql = " select * from quiz  ";
        	 	try (PreparedStatement pstmt = conn.prepareStatement(sql)){
        	 		ResultSet resultSet =  pstmt.executeQuery();
        	 		while(resultSet.next()) {
        	 			System.out.println("문제 ID : " + resultSet.getInt("id"));
        	 			System.out.println("문제 : " + resultSet.getString("question"));
        	 			System.out.println("정답 : " + resultSet.getString("answer"));
        	 			if(!resultSet.isLast()) {
        	 				System.out.println("----------------------------------------");
        	 			}
        	 		}
        		} catch (SQLException e) {
        			e.printStackTrace();
        		}
        	}
        
        	private static void addQuizQuestion(Connection conn, Scanner scanner) {
        		
        		System.out.println("퀴즈 문제를 입력하세요: ");
        		scanner.nextLine();
        		String question = scanner.nextLine();
        		System.out.println("퀴즈 정답을 입력하세요: ");
        		String answer = scanner.nextLine();
        		
        		String sql = " insert into quiz(question, answer) values(?, ?) ";
        		
        		try (PreparedStatement pstmt = conn.prepareStatement(sql)){
        			pstmt.setString(1, question);
        			pstmt.setString(2, answer);
        			int rowsInsertedCount = pstmt.executeUpdate();
        			System.out.println("추가된 행의 수 : " + rowsInsertedCount);
        		} catch (SQLException e) {
        			e.printStackTrace();
        		}
        	}
        }  // end of class 
        

        코드 리팩토링 - 1단계 DB 연결을 처리하는 클래스를 따로 분리하면 재사용성과 유지보수성이 높아집니다

         

        1단계 핵심

        DBConnectionManager 클래스 만들어 보기 데이터베이스 연결을 관리하는 DBConnectionManager 클래스를 만들어 봅시다.

        package com.tenco.quiz;
        
        import java.sql.Connection;
        import java.sql.DriverManager;
        import java.sql.SQLException;
        
        public class DBConnectionManager {
        	
        	private static final String URL = "jdbc:mysql://localhost:3306/quizdb?serverTimezone=Asia/Seoul";
        	private static final String USER = "root";
        	private static final String PASSWORD = "asd123";
        	
        	// static {} 블록 - 정적 초기화 블록 
        	// 클래스가 처음 로드될 때 한 번 실행 됩니다. 
        	// 정적 변수의 초기화나 복잡한 초기화 작업을 수행할 때 사용 
        	// static {} 블록안에 예외를 던질 수도 있다. 
        	static {
        		try {
        			Class.forName("com.mysql.cj.jdbc.Driver");
        		} catch (ClassNotFoundException e) {
        			e.printStackTrace();
        		}
        	}
        	
        	// 정적 메서드(함수) 커넥션 객체를 리턴하는 함수를 만들어 보자
        	public static Connection getConnection() throws SQLException {
        		return DriverManager.getConnection(URL, USER, PASSWORD);
        	}
        	
        } // end of class 
        
        

        사용

        package com.tenco.quiz.ver1;
        
        import java.sql.Connection;
        import java.sql.DriverManager;
        import java.sql.PreparedStatement;
        import java.sql.ResultSet;
        import java.sql.SQLException;
        import java.util.Scanner;
        
        import com.tenco.quiz.DBConnectionManager;
        
        public class QuizGame {
        	
        	private static final String ADD_QUIZ = " insert into quiz(question, answer) values(?, ?) "; 
        	private static final String VIEW_QUIZ = " select * from quiz "; 
        	private static final String RANDOM_QUIZ = " select * from quiz order by rand() limit 1 "; 
        	
        	
        	public static void main(String[] args) {
        		
        		try (Connection conn = DBConnectionManager.getConnection();
        			  Scanner scanner = new Scanner(System.in)) {
        			
        			while(true) {
        				printMenu();
        				int choice = scanner.nextInt(); // 블로킹 
        				
        				if(choice == 1) {
        					addQuizQuestion(conn, scanner);
        				} else if(choice == 2) {
        					viewQuizQuestion(conn);
        				} else if(choice == 3) {
        					playQuizGame(conn, scanner);
        				} else if(choice == 4) {
        					System.out.println("프로그램을 종료 합니다");
        					break;
        				} else  {
        					System.out.println("잘못된 선택입니다. 다시 시도하세요.");
        				}
        			}
        		} catch (Exception e) {
        			e.printStackTrace();
        		}
        	} // end of main
        
        	private static void printMenu() {
        		System.out.println();
        		System.out.println("----------------------------------------");
        		System.out.println("1. 퀴즈 문제 추가");
        		System.out.println("2. 퀴즈 문제 조회");
        		System.out.println("3. 퀴즈 게임 시작");
        		System.out.println("4. 종료");
        		System.out.print("옵션을 선택 하세요: ");
        	}
        
        	private static void playQuizGame(Connection conn, Scanner scanner) {
        		
        		try (PreparedStatement pstmt = conn.prepareStatement(RANDOM_QUIZ)) {
        			ResultSet rs = pstmt.executeQuery();
        			
        			if(rs.next()) {
        				
        				String question = rs.getString("question");
        				String answer = rs.getString("answer");
        				
        				System.out.println("퀴즈 문제 : " + question);
        				// 버그처리 
        				scanner.nextLine();
        				System.out.print("당신에 답: ");
        				String userAnswer = scanner.nextLine();
        				
        				if(userAnswer.equalsIgnoreCase(answer)) {
        					System.out.println("정답입니다! 점수를 얻었습니다.");
        				} else {
        					System.out.println("오답입니다!");
        					System.out.println("퀴즈 정답은 -->  " + answer);
        				}
        			} else {
        				System.out.println("죄송합니다 아직 퀴즈 문제를 만들고 있습니다.");
        			}
        		} catch (SQLException e) {
        			e.printStackTrace();
        		}
        	}
        
        	private static void viewQuizQuestion(Connection conn) {
        		
        	 	try (PreparedStatement pstmt = conn.prepareStatement(VIEW_QUIZ)){
        	 		ResultSet resultSet =  pstmt.executeQuery();
        	 		while(resultSet.next()) {
        	 			System.out.println("문제 ID : " + resultSet.getInt("id"));
        	 			System.out.println("문제 : " + resultSet.getString("question"));
        	 			System.out.println("정답 : " + resultSet.getString("answer"));
        	 			if(!resultSet.isLast()) {
        	 				System.out.println("----------------------------------------");
        	 			}
        	 		}
        		} catch (SQLException e) {
        			e.printStackTrace();
        		}
        	}
        
        	private static void addQuizQuestion(Connection conn, Scanner scanner) {
        		
        		System.out.println("퀴즈 문제를 입력하세요: ");
        		scanner.nextLine();
        		String question = scanner.nextLine();
        		System.out.println("퀴즈 정답을 입력하세요: ");
        		String answer = scanner.nextLine();
        				
        		try (PreparedStatement pstmt = conn.prepareStatement(ADD_QUIZ)){
        			pstmt.setString(1, question);
        			pstmt.setString(2, answer);
        			int rowsInsertedCount = pstmt.executeUpdate();
        			System.out.println("추가된 행의 수 : " + rowsInsertedCount);
        		} catch (SQLException e) {
        			e.printStackTrace();
        		}
        	}
        }  // end of class 
        

        코드 리팩토링 - 2단계 QuizGame을 SOLID 원칙에 따라 리팩토링해보기

        1. 단일 책임 원칙 (Single Responsibility Principle, SRP): 클래스는 하나의 책임만 가져야 합니다.
        2. 개방-폐쇄 원칙 (Open/Closed Principle, OCP): 소프트웨어 개체는 확장에는 열려 있어야 하지만, 수정에는 닫혀 있어야 합니다.
        3. 리스코프 치환 원칙 (Liskov Substitution Principle, LSP): 프로그램의 객체는 프로그램의 정확성을 깨뜨리지 않으면서 하위 타입의 인스턴스로 바꿀 수 있어야 합니다.
        4. 인터페이스 분리 원칙 (Interface Segregation Principle, ISP): 특정 클라이언트를 위한 인터페이스 여러 개가 범용 인터페이스 하나보다 낫다.
        5. 의존성 역전 원칙 (Dependency Inversion Principle, DIP): 고수준 모듈은 저수준 모듈에 의존해서는 안되며, 둘 다 추상화에 의존해야 한다.
        package com.tenco.quiz;
        
        import java.sql.Connection;
        import java.sql.DriverManager;
        import java.sql.SQLException;
        
        public class DBConnectionManager {
        	
        	private static final String URL = "jdbc:mysql://localhost:3306/quizdb?serverTimezone=Asia/Seoul";
        	private static final String USER = "root";
        	private static final String PASSWORD = "asd123";
        	
        	// static {} 블록 - 정적 초기화 블록 
        	// 클래스가 처음 로드될 때 한 번 실행 됩니다. 
        	// 정적 변수의 초기화나 복잡한 초기화 작업을 수행할 때 사용 
        	// static {} 블록안에 예외를 던질 수도 있다. 
        	static {
        		try {
        			Class.forName("com.mysql.cj.jdbc.Driver");
        		} catch (ClassNotFoundException e) {
        			e.printStackTrace();
        		}
        	}
        	
        	// 정적 메서드(함수) 커넥션 객체를 리턴하는 함수를 만들어 보자
        	public static Connection getConnection() throws SQLException {
        		
        		new Thread().run();
        		
        		return DriverManager.getConnection(URL, USER, PASSWORD);
        	}
        	
        } // end of class 
        
        
        package com.tenco.quiz.ver2;
        
        import java.sql.SQLException;
        import java.util.List;
        
        public interface QuizRepository {
        	
        	int addQuizQuestion(String question, String answer) throws SQLException ;
        	List<QuizDTO> viewQuizQuestion() throws SQLException;
        	QuizDTO playQuizGame() throws SQLException;
        }
        
        package com.tenco.quiz.ver2;
        
        import lombok.AllArgsConstructor;
        import lombok.Getter;
        import lombok.NoArgsConstructor;
        import lombok.Setter;
        import lombok.ToString;
        
        @Getter
        @Setter
        @NoArgsConstructor
        @AllArgsConstructor
        @ToString
        public class QuizDTO {
        	
        	private int id; 
        	private String question; 
        	private String answer; 
        	
        }
        
        
        package com.tenco.quiz.ver2;
        
        import java.sql.Connection;
        import java.sql.PreparedStatement;
        import java.sql.ResultSet;
        import java.sql.SQLException;
        import java.util.ArrayList;
        import java.util.List;
        
        import com.tenco.quiz.DBConnectionManager;
        
        public class QuizRepositoryImpl implements QuizRepository {
        	
        	public static final String ADD_QUIZ = " insert into quiz(question, answer) values(?, ?) "; 
        	public static final String VIEW_QUIZ = " select * from quiz "; 
        	public static final String RANDOM_QUIZ = " select * from quiz order by rand() limit 1 "; 
        	
        	@Override
        	public int addQuizQuestion(String question, String answer) throws SQLException {
        		int resultRowCount = 0; 
        		try (Connection conn = DBConnectionManager.getConnection()){
        			PreparedStatement pstmt = conn.prepareStatement(ADD_QUIZ);
        			// 트랜잭션 처리 가능 - 수동모드 커밋 사용 가능 
        			pstmt.setString(1, question);
        			pstmt.setString(2, answer);
        			pstmt.executeUpdate();
        		} 
        		return resultRowCount;
        	}
        
        	@Override
        	public List<QuizDTO> viewQuizQuestion() throws SQLException {
        		List<QuizDTO> list = new ArrayList<>();
        		try(Connection conn = DBConnectionManager.getConnection()) {
        			PreparedStatement pstmt = conn.prepareStatement(VIEW_QUIZ);
        			ResultSet rs = pstmt.executeQuery();
        			while(rs.next()) {
        				int id = rs.getInt("id");
        				String question = rs.getString("question");
        				String answer = rs.getString("answer");
        				list.add(new QuizDTO(id, question, answer))  ;
        			}
        		}
        		return list;
        	}
        
        	@Override
        	public QuizDTO playQuizGame() throws SQLException {
        		QuizDTO quizDTO = null;
        		try (Connection conn = DBConnectionManager.getConnection()) {
        			PreparedStatement pstmt = conn.prepareStatement(RANDOM_QUIZ);
        			ResultSet rs = pstmt.executeQuery();
        			if(rs.next()) {
        				int id = rs.getInt("id");
        				String question = rs.getString("question");
        				String answer = rs.getString("answer");
        				quizDTO = new QuizDTO(id, question, answer);
        			}
        		} 
        		return quizDTO;
        	}
        }
        
        

         

        package com.tenco.quiz.ver2;
        
        public class QuizRepsoitoryTest1 {
        
        	public static void main(String[] args) {
        		
        		// 메서드 호출해서 실행 확인 디버깅 확인 테스트 반드시 한다 !! 
        		// QuizRepsoitory 구현 클래스 테스트 
        		
        		// 주말 과제 
        		// 실행에 흐름 여러분 직접 만들어 보기 
        	}
        
        }

        'Data Structure( 자료구조 ) > JDBC' 카테고리의 다른 글

        JDBC에서의 예외 처리  (1) 2024.09.16
        JDBC 성능 최적화  (0) 2024.09.16
        JDBC 트랜잭션 관리와 배치 처리  (0) 2024.09.16
        JDBC 기본 사용법  (0) 2024.09.16
        JDBC설치 및 설정  (0) 2024.09.16
        다음글
        다음 글이 없습니다.
        이전글
        이전 글이 없습니다.
        댓글
      조회된 결과가 없습니다.
      스킨 업데이트 안내
      현재 이용하고 계신 스킨의 버전보다 더 높은 최신 버전이 감지 되었습니다. 최신버전 스킨 파일을 다운로드 받을 수 있는 페이지로 이동하시겠습니까?
      ("아니오" 를 선택할 시 30일 동안 최신 버전이 감지되어도 모달 창이 표시되지 않습니다.)
      목차
      표시할 목차가 없습니다.
        • 안녕하세요
        • 감사해요
        • 잘있어요

        티스토리툴바