[ITWILL : JSP]Javabean 6 : 게시판만들기(글쓰기)

ITWILL학원 : 27강 JSP기초 BY 정규태강사

1. 웹 개발의 기초 : CRUD

https://newstars.tistory.com/436

  • 기본 로직 :

    1. 게시판 글쓰기 = 데이터입력
    2. 글 리스트 = 데이터검색
    3. 글 본문보기 = 데이터검색
    4. 글 수정 = 데이터 수정
    5. 글 삭제 = 데이터 삭제
  • 추가할 로직 :

    1. 답글쓰기 (댓글과 다른 답글기능)
    2. 파일업로드 / 썸네일

위와 같은 두 개의 로직은 요구사항분석이라고도 부른다

1-1. 개발순서

  1. 사용자 요구사항 분석
  2. 데이터 테이블 설계 = DB
    1. 테이블생성 테이블명 itwill_board
    2. 글번호 : bno
    3. 글쓴이 : name
    4. 글 비밀번호 : pw
    5. 글 제목 : subject
    6. 글 내용 : content
    7. 조회수 : readcount
    8. 답글 : re_ref, re_lev, re_seq
    9. 글쓴날짜 : date
    10. 파일 : file
      • varchar(200)를 쓰는 이유는
    11. 글쓴이의 IP주소 : ip
  3. writeForm.jsp 작성
  4. writePro.jsp 작성
  5. BoardBean.java 생성
  6. BoardDAO.java 에서 getCon() DB연결메서드 구현
  7. BoardDAO.java 에서 insertBoard() 글쓰기메서드 구현

1-2. 데이터 테이블 설계 = DB

테이블생성 테이블명 itwill_board을 생성한 뒤 아래처럼 컬럼을 작성한다.

  1. 글번호 : bno - INT NOT NULL, PRIMARY KEY (bno)
    • 글번호는 AI(Auto Increment)를 해야한다 말아야한다는 의견이 분분하다. 실습에선 AI없이 진행.
  2. 글쓴이 : name - VARCHAR(45) NOT NULL
  3. 글 비밀번호 : pw - VARCHAR(45) NOT NULL
  4. 글 제목 : subject - VARCHAR(45) NOT NULL
  5. 글 내용 : content - VARCHAR(2000) NOT NUL
  6. 조회수 : readcount - INT NULL
  7. 답글 : re_ref, re_lev, re_seq - 세 컬럼 다 INT NULL
  8. 글쓴날짜 : date - DATE NULL
  9. 파일 : file - VARCHAR(200) NULL
    • varchar(200)를 쓰는 이유는 파일은 DB가 아닌 서버에 저장되고 DB는 이름만 저장해놓기때문.
  10. 글쓴이의 IP주소 : ip - VARCHAR(45) NULL

1-3. writeForm.jsp 작성

1
2
3
4
5
6
7
8
9
10
11
12
<fieldset>
<legend>게시판 글쓰기</legend>
<form action="writePro.jsp" method="post" name="fr">
글쓴이 : <input type="text" name="name"><br>
비밀번호 : <input type="password" name="pw"><br>
제목 : <input type="text" name="subject"><br>
내용 : <br>
<textarea rows="10" cols="35" name="content" placeholder="여기에 작성해주세요"></textarea><br>
<input type="submit" value="글쓰기">
<button type="reset">초기화</button>
</form>
</fieldset>

1-4. writePro.jsp 작성

글쓰기를 처리하는 페이지이다.
입력창에서 받은 데이터를 DB로 이동시킨 뒤 사용자는 글목록페이지로 이동

  • jsp:setProperty 의 동작방식

      1. String name = request.getParameter(“bno”)
      1. bb.setbno(bno)
    • 모든 변수에 위의 1번,2번동작을 처리
  • 전달정보 확인위한 출력

    • System.out.println(bb)
    • System.out.println(bb.toString())
    • 위의 두 코드는 동일한 출력값을 가진다.
  • 로그인한사람만 글쓸수있게 처리

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
<%//1. 한글처리
request.setCharacterEncoding("UTF-8");
%>

//6. 로그인한사람만 글쓸수있게처리
String id = (String) session.getAttribute("id");
if( id == null){
%>
<script type="text/javascript">
alert("로그인 후 이용가능합니다");
history.back();
</script>
<%
}

<!-- 2. 액션태그사용 : 전달된 정보를 저장(bno, name, pw 등등) - 자바빈객체에 저장-->
<jsp:useBean id="bb" class="com.itwillbs.board.BoardBean" />
<!-- 2-1. 파라미터 값 저장 : jsp:setProperty -->
<jsp:setProperty property="*" name="bb"/>
<%//2-2. 전달정보 확인위한 출력 : null인 값들은 별로도 추가해줘야한다.
System.out.println(bb); //bb.toString()과 동일한 동작
//2-3. IP정보를 추가
System.out.println(request.getRemoteAddr());
bb.setIp(request.getRemoteAddr());
System.out.println(bb);
//2-4. date정보

//3. 디비처리작업진행 : boardDAO객체 생성
BoardDAO bdao = new BoardDAO();

//4. 데이터처리 : BoardDAO객체 안의 메서드 중 글스기 기능을 하는 메서드 호출
bdao.insertBoard(bb);

// 5. 글목록 페이지로 이동
response.sendRedirect("boardList.jsp");

1-5. BoardBean.java 생성

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
//1번규칙만족 : 클래스는 public
public class BoardBean {
//2번규칙만족 : 멤버변수선언 private
private int bno;
private String name;
private String pw;
private String subject;
private String content;
private int readcount;
private int re_ref;
private int re_lev;
private int re_seq;
private Date date;
private String file;
private String ip;

//4번규칙만족 : 기본생성자존재하지만 생략됨
//public BoardBean(){}

//3번규칙만족 : 멤버변수마다 별도의 get/set메소드가 존재해야한다.
public int getBno() {
return bno;
}
public void setBno(int bno) {
this.bno = bno;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}

//(중략)

public String getIp() {
return ip;
}
public void setIp(String ip) {
this.ip = ip;
}

//5. toString()
@Override
public String toString() {
return "BoardBean [bno=" + bno + ", name=" + name + ", pw=" + pw + ",
subject=" + subject + ", content=" + content + ", readcount=" + readcount + ",
re_ref=" + re_ref + ", re_lev=" + re_lev + ", re_seq=" + re_seq + ",
date=" + date + ", file=" + file + ", ip=" + ip + "]";
}

1-6. BoardDAO.java 에서 getCon() DB연결메서드 구현

드라이브로드, DB연결 메서드인 getCon()구현

  • getCon() 작성순서 : 리턴값없이 진행

      1. 라이브러리설치 (WEB-INF, LIB에 추가)
      1. /META-INF/context.xml 파일생성
      1. /WEB-INF/web.xml 파일수정
      1. DAO 처리
  • 예외처리 throws 방법 : 메서드를 호출하는 시점에 예외처리를 함

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
public class BoardDAO {
Connection con = null;
String sql = "";
PreparedStatement pstmt = null;
ResultSet rs = null;

//DB연결메서드 구현
private void getCon() throws Exception{
//Context 객체 생성
//Context인터페이스이기때문에 직접객체생성할 수 없어서 InitialContext클래스를 사용해서 객채생성
//예외처리 : throws사용하는 방법
Context init = new InitialContext(); //업캐스팅
//디비연결정보를 불러오기 ->DataSource 타입으로 저장
// 고정문구"java:comp/env/다른문구context파일의 name값입력"
DataSource ds = (DataSource) init.lookup("java:comp/env/jdbc/mysqlDB");
//ds 사용해서 연결
//멤버변수는 static이 아니기때문에 new BoardDAO();객체생성 후에 멤버변수들이 생성된다.
//그리고 getCon()메서드를 하면 멤버변수 con에 데이터(ds.getConnection();)가 담기게 된다.
//따라서 이때 return하지 않고 써도된다.
con = ds.getConnection();
System.out.println("디비연결성공 + con");
}//getCon닫음

//자원해제 메서드 구현
public void closeDB(){
try{
if(rs != null) rs.close();
if(pstmt != null) pstmt.close();
if(con != null) con.close();
} catch (SQLException e) {
e.printStackTrace();
}
}//closeDB닫음

}BoardDAO클래스닫음

1-7. BoardDAO.java 에서 insertBoard() 글쓰기 메서드 구현

insertBoard()메서드 구현

  • if(rs.next())가 끝나는 종료되는 순간은?
    • 글이 없으면 null인데 어떻게 조건문이 true가 되어 구현될수있을까?
    • rs가 faluse가 되는 시점은 EOF만나는 순간이다.
    • 즉, NULL이 있어도 진행된다.
  • getInt()사용법 두가지
    • getInt(컬럼명)사용 : num = rs.getInt(“max(bno)”)+1;
    • getInt(idx)사용 : 둘다 동일한 결과. idx사용하면 컬럼명보다 데이터 처리속도도 빠르다- /getInt => 데이터 값을 리턴, 값이 SQL-NULL경우 0리턴
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
//글쓰기 메서드 구현
public void insertBoard(BoardBean bb){
//글번호를 저장하는 변수 생성
int num = 0;

try {
//1. 디비연결
getCon();
//2. SQL구문작성 & pstmt
//3. 글번호계산 : 지금 있는 글 번호중 가장 큰 번호가 뭔지 확인하기만 하면 되니까
sql = "select max(bno) from itwill_board";
pstmt = con.prepareStatement(sql);
//4. 글번호계산한거 실행
rs = pstmt.executeQuery();

//5. 데이터처리(글번호 계산)
if(rs.next()){
//글이 없으면 null이 어떻게 조건문을 돌수있을까? rs가 faluse가 되는 시점은 EOF만나는 순간이다.
//즉, NULL이 있어도 진행된다.

//getInt(컬럼명)사용 : num = rs.getInt("max(bno)")+1;
//getInt(idx)사용 : 둘다 동일한 결과. idx사용하면 컬럼명보다 데이터 처리속도도 빠르다
//getInt => 데이터 값을 리턴, 값이 SQL-NULL경우 0리턴
num = rs.getInt(1)+1;
}

System.out.println("글번호 : "+num);
//6. 글쓰기(저장) : 실무에서는 물음표가 기본 20개이상이다. 5개씩 잘라서 String연결하면 헷갈리지않는다.
//Date는 now()메서드를 이용한다.
//now() : 자동으로 SQL구문 실행시 시스템시간 정보를 불러옴
sql = "insert into itwill_board value("
+"?,?,?,?,?,"
+"?,?,?,?,now(),"
+"?,?)";
pstmt = con.prepareStatement(sql);
pstmt.setInt(1, num);//bno는 가지고있는게 아니라 계산해서 만든것이므로 num사용
pstmt.setString(2, bb.getName());
pstmt.setString(3, bb.getPw());
pstmt.setString(4, bb.getSubject());
pstmt.setString(5, bb.getContent());
pstmt.setInt(6, 0);//조회수는 글쓰지마자 올라가지않는다. 글쓰는 당시에는 조회수가 0이니까 초기화한다.
pstmt.setInt(7, num);//답변글 그룹번호이므로 일반글번호와 동일하다.
pstmt.setInt(8, 0);//답변글 들여쓰기(처음 작성하는 일반 글의 들여쓰기는 0이므로 초기화한다)
pstmt.setInt(9, 0);//답변글순서(처음 작성하는 일반글이므로 일반글 제일위쪽(0)으로 초기화한다)
pstmt.setString(10, bb.getFile());
pstmt.setString(11, bb.getIp());

//7. 실행
pstmt.executeUpdate();
System.out.println("게시판글쓰기 성공");
} catch (Exception e) {
// TODO Auto-generated catch block
System.out.println("게시판글쓰기 실패");
e.printStackTrace();
} finally {
//자원해제
closeDB();
}
} //insertBoard메서드닫음

2. 예외처리 차이점(try catch와 throws)

  • try catch 방법
  • throws 방법

자바 예외처리(try catch, throws)
try catch는 예외를 직접 처리하기 때문에 메서드를 호출해도 더 이상 예외처리를 할 필요가 없습니다.
하지만 throws같은 경우는 호출하는 쪽에다가 예외처리를 맡겨버리는(?) 느낌?
그래서 호출하는 메소드에 throws가 있으면
호출하는 쪽에서도 예외처리를 다시 해줘야되요!