• 티스토리 홈
  • 프로필사진
    Song hyun
  • 방명록
  • 공지사항
  • 태그
  • 블로그 관리
  • 글 작성
Song hyun
  • 프로필사진
    Song hyun
    • 분류 전체보기 (780)
      • 백준 (0)
      • 영어 (2)
        • Diary (0)
        • Toast Masters (2)
      • 메모 (13)
      • 설치 메뉴얼 (30)
      • Java (178)
      • MySQL (60)
      • JSP (67)
      • Springboot (46)
      • HTML,CSS, JS (71)
        • HTML (8)
        • CSS (12)
        • JavaScript (37)
        • HTML&CSS 스터디 (13)
      • C++ (7)
      • Linux (7)
      • JPA (34)
      • Kotlin (2)
      • Flutter (42)
      • Error Note (39)
      • 디자인 패턴 (12)
      • 디지털논리회로 (4)
      • 데이터베이스 시스템 (8)
      • 알고리즘 (7)
      • 운영체제 (3)
      • 이산수학 (3)
      • 인공지능 (1)
      • 자료 구조 (14)
        • 기본 개념 (14)
        • 자료구조 스터디 (0)
      • 💡My project (76)
        • 팩맨 : Java Swing 게임 제작 프로젝트 (6)
        • 네이트톡 : Java 소켓 통신 프로젝트 (4)
        • 포켓옥션 : HikariCP&JDBC CRUD 프.. (3)
        • 이지 부산 : BDIA-Devton 2024 프로.. (20)
        • 그린 유니버시티 : JSP를 사용한 학사관리 프로.. (1)
        • 애드 포커 : 웹 소켓과 Spring을 사용한 카.. (1)
        • 셸위 : 게임 친구 매칭 사이트 (21)
        • 다모아 : 개발자 중개 플랫폼 (20)
      • 📗스터디 (13)
        • CNN : 웹개발 스터디 (10)
        • Node&React로 유튜브 사이트 만들기 (3)
      • 📙독서 및 강연 기록 (36)
        • 강연 (14)
        • 독서 (22)
  • 방문자 수
    • 전체:
    • 오늘:
    • 어제:
  • 최근 댓글
      등록된 댓글이 없습니다.
    • 최근 공지
        등록된 공지가 없습니다.
      # Home
      # 공지사항
      #
      # 태그
      # 검색결과
      # 방명록
      • [Java] 121. JDBC를 활용한 CRUD와 SOLID 원칙 (2)
        2024년 06월 14일
        • Song hyun
        • 작성자
        • 2024.06.14.오전11:20
        728x90
        반응형

        [Java] 121. JDBC를 활용한 CRUD와 SOLID 원칙 (2)

        콘솔을 활용한 퀴즈 게임 만들기 - 리팩토링하기
        *https://whatsthatsound.tistory.com/337 에서 작성한 코드를 리팩토링해보자.
         
        * 리팩토링= 코드를 보기 좋게 정리하는 것 or 성능을 향상시키는 것.
        ** DB 연결을 처리하는 클래스를 따로 분리하면, 재사용성과 유지보수성이 높아진다


        (1) 리팩토링 1단계 : 클래스를 분리한다. (=DBConnectionManager)

        *static {} 블록 - 정적 초기화 블록:
        /클래스가 처음 로드될 때 한 번 실행된다. 정적 변수의 초기화나 복잡한 초기화 작업을 수행할 때 사용되며, static{} 블록 안에 예외를 던질 수도 있다.
         

        • 메서드는 따로 분리
        • 자주 쓰이는 객체는 클래스화

        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;
        
        import com.mysql.cj.x.protobuf.MysqlxPrepare.Execute;
        
        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) {
        					// 퀴즈 문제 추가
        					// 사용자 퀴즈와 답을 입력받아야 함
        					// connection을 활용해서 query를 날려야 한다.
        					addQuizQuestion(conn, scanner);
        
        				} else if (choice == 2) {
        					// 퀴즈 문제 조회
        					viewQuizQuestion(conn);
        					
        				} else if (choice == 3) {
        					// 퀴즈 게임 시작
        					playQuizQuestion(conn,scanner);
        					
        				} else if (choice == 4) {
        					// 게임 종료
        					System.out.println("게임을 종료합니다.");
        					return;
        					
        				} 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.println("옵션을 선택하세요.");
        	}
        
        
        	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();
        		}
        
        	}
        	
        	private static void viewQuizQuestion(Connection conn) {
        		try (PreparedStatement pstmt= conn.prepareStatement(VIEW_QUIZ);){
        			ResultSet resultSet = pstmt.executeQuery(); // ResultSet을 반환한다.
        			while(resultSet.next()) { // 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) {
        			// TODO Auto-generated catch block
        			e.printStackTrace();
        		}
        		
        	}
        	
        
        	private static void playQuizQuestion(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();
        		}
        		
        		
        	}
        	
        	
        
        }
        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?serverTimezon=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);
        	}
        	
        }

        (2) 리팩토링 2단계 : SOLID 원칙에 따라 리팩토링 해보기

         
        *SOLID 원칙이란?

        • 단일 책임 원칙 (Single Responsibility Principle/SRP): 클래스는 하나의 책임만 가져야 한다.
        • 개방-폐쇄 원칙 (Opne-Closed Principle/OCP): 소프트웨어 개체는 확장에는 열려 있어야 하지만, 수정에는 닫혀 있어야 한다.
        • 리스코프 치환 원칙 (Liskov Substitution Principle/LSP): 프로그램의 객체는 프로그램의 정확성을 깨뜨리지 않으면서, 하위 타입의 인스턴스로 바꿀 수 있어야 한다.
        • 인터페이스 분리 원칙 (Interface Segregation Principle/ISP): 특정 클라이언트를 위한 인터페이스 여러 개가 범용 인터페이스 하나보다 낫다.
        • 의존성 역전 원칙 (Dependency Inversion Principle/DIP): 고수준 모듈은 저수준 모듈에 의존해서는 안 되며, 둘 다 추상화에 의존해야 한다.

         
        1. 추상화하기 (인터페이스)

        게임의 기능을 인터페이스화(좌측) 퀴즈를 클래스화(우측)

         

         
         
        (1) QuizRepository (인터페이스)

        package com.tenco.quiz.ver2;
        
        import java.sql.Connection;
        import java.sql.SQLException;
        import java.util.List;
        import java.util.Scanner;
        
        public interface QuizRepository {
        	
        	int addQuizQuestion(String question, String answer) throws SQLException;
        	// todo 수정 예정
        	List<QuizDTO> viewQuizQuestion()throws SQLException;
        	QuizDTO playQuizQuestion()throws SQLException;
        
        }

         
        (2) QuizDTO (구현클래스)

        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;
        
        }

         
        (3) QuizRepository (인터페이스 상속 클래스)

        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.ver1.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));
        			}
        		} catch (Exception e) {
        			// TODO: handle exception
        		}
        		
        		return list;
        	}
        
        	@Override
        	public QuizDTO playQuizQuestion() 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);
        			}
        		} catch (Exception e) {
        		}
        		return quizDTO;
        	}
        
        }

         
        (4) MainTest (시범 클래스)

        package com.tenco.quiz.ver2;
        
        import java.sql.SQLException;
        import java.util.List;
        
        public class MainTest1 {
        	
        	public static void main(String[] args) {
        		QuizRepositoryImpl quizRepositoryImpl=new QuizRepositoryImpl();
        		try {
        //			List<QuizDTO> quizDtos = quizRepositoryImpl.viewQuizQuestion();
        //			for(QuizDTO quizDTO : quizDtos) {
        //				System.out.println(quizDTO);
        //			}
        			QuizDTO dto=quizRepositoryImpl.playQuizQuestion();
        			System.out.println(dto);
        			System.out.println("정답을 맞춰주세요");
        			System.out.println(dto.getQuestion());
        			String userInput="한국";
        			if(dto.getAnswer().equalsIgnoreCase(userInput)) {
        				System.out.println("정답입니다");
        			} else {
        				System.out.println("오답");
        			}
        		} catch (SQLException e) {
        			e.printStackTrace();
        		}
        		
        	}
        
        }
        728x90
        반응형

        'Java' 카테고리의 다른 글

        [입문 문제] 3. 나눗셈 - 몫 구하기  (0) 2024.06.15
        [입문 문제] 2. 두 수의 차 구하기  (0) 2024.06.14
        [Java] 120. JDBC를 활용한 CRUD와 SOLID 원칙 (1)  (0) 2024.06.14
        [입문 문제] 1. 두 수의 곱 구하기  (0) 2024.06.13
        [Java] 119. 트랜잭션과 배치 처리  (0) 2024.06.13
        다음글
        다음 글이 없습니다.
        이전글
        이전 글이 없습니다.
        댓글
      조회된 결과가 없습니다.
      스킨 업데이트 안내
      현재 이용하고 계신 스킨의 버전보다 더 높은 최신 버전이 감지 되었습니다. 최신버전 스킨 파일을 다운로드 받을 수 있는 페이지로 이동하시겠습니까?
      ("아니오" 를 선택할 시 30일 동안 최신 버전이 감지되어도 모달 창이 표시되지 않습니다.)
      목차
      표시할 목차가 없습니다.
      • [Java] 121. JDBC를 활용한 CRUD와 SOLID 원칙 (2)
      • 안녕하세요
      • 감사해요
      • 잘있어요

      티스토리툴바

      단축키

      내 블로그

      내 블로그 - 관리자 홈 전환
      Q
      Q
      새 글 쓰기
      W
      W

      블로그 게시글

      글 수정 (권한 있는 경우)
      E
      E
      댓글 영역으로 이동
      C
      C

      모든 영역

      이 페이지의 URL 복사
      S
      S
      맨 위로 이동
      T
      T
      티스토리 홈 이동
      H
      H
      단축키 안내
      Shift + /
      ⇧ + /

      * 단축키는 한글/영문 대소문자로 이용 가능하며, 티스토리 기본 도메인에서만 동작합니다.