3
1

More than 1 year has passed since last update.

【JSP/Servlet】簡単なチャット機能の作り方

Last updated at Posted at 2022-07-17

はじめに

初学者向けのJavaサーブレットを用いた、もっとも原始的なチャット機能の実装方法です。
個人での開発やスクールでの課題作成時に参考にしてください。

開発環境

カテゴリー ソフトウェア名 バージョン
OS Windows10 21H2
エディタ Eclipse 2018 4.8.0
データベース MySQL 5.5.39

設計

ER図
image.png

DB設計書

  • ユーザ管理テーブル
  • メッセージ管理テーブル
  • チャットルーム管理テーブル

データベースは、以上の3つに分かれています。

① ユーザ管理テーブル
image.png

CREATE TABLE accountinfo(
account_id VARCHAR(50) PRIMARY KEY NOT NULL,
password VARCHAR(20) NOT NULL
)ENGINE=InnoDB;

② メッセージ管理テーブル
image.png

CREATE TABLE messageinfo(
message_id INTEGER PRIMARY KEY NOT NULL AUTO_INCREMENT,
room_id INTEGER,
account_id VARCHAR(50),
message VARCHAR(300),
creation_day TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
FOREIGN KEY(room_id) REFERENCES roominfo(room_id) on update cascade on delete cascade,
FOREIGN KEY(account_id) REFERENCES accountinfo(account_id) on update cascade on delete cascade
)ENGINE=InnoDB;

③ チャットルーム管理テーブル
image.png

CREATE TABLE roominfo(
room_id INTEGER AUTO_INCREMENT PRIMARY KEY NOT NULL,
account_id VARCHAR(50),
FOREIGN KEY(account_id) REFERENCES accountinfo(account_id) ON UPDATE CASCADE ON DELETE CASCADE
)ENGINE=InnoDB;

JSP

仕組みはとても単純で
iframeを用いて、チャットを表示している画面を埋め込むというものです。

  • messageroom.jsp(メイン画面 )
  • message.jsp(チャットのみ表示)

messageroom.jsp
image.png

<%@page contentType="text/html;charset=UTF-8"%>
<!-- ルームIDをリクエストスコープから取得 -->
<%String roomid =  (String) request.getAttribute("roomid");%>

<html>
<head>
<title>チャット</title>
<link rel="stylesheet" type="text/css" href="<%=request.getContextPath()%>/css/stylesheet.css">
</head>
<body>
	<div class="item-wrapper">
		<div class="container">
			<!-- チャット画面を埋め込む -->
			<iframe src="<%= request.getContextPath()%>/message?roomid=<%=roomid%>" width="650px" height="500px margin:0 auto;"></iframe>
			<!-- メッセージ送信用の入力欄 -->
			<form method="post" action="<%=request.getContextPath()%>/message?roomid=<%=roomid%>&cmd=1">
			<p>
				<input type=text size="70" name="message" placeholder="メッセージを入力してください"></input>
				<input type="submit" value="送信"></input>
			</p>
			</form>
		</div>
	</div>
</body>
</html>

message.jsp
image.png

<%@page contentType="text/html;charset=UTF-8"%>
<%@page import="java.util.ArrayList,bean.User,bean.Message"%>

<%
//メッセージリストを読み込み
ArrayList<Message> messageList = (ArrayList<Message>)request.getAttribute("messageList");
User userinfo = (User)request.getAttribute("userinfo");
%>

<html>
<head>
<meta charset="utf-8">
<title>メッセージ</title>
</head>
<body style="background-color:#ffffff">
<div style="width:100%;">
<%if (messageList.size() != 0) {
	for (Message message:messageList) { %>
	<% if (message.getSenderId().equals(userinfo.getUser_id())) { %>
		<div style="margin-left: 0px; text-align: right;">
			<p><%=userinfo.getUser_name() %></p>
			<p><%=message.getMessage() %></p>
		</div>
	<%} else { %>
		<div style="margin-right: 0px;">
			<p><%=message.getSenderId() %></p>
			<p><%=message.getMessage() %></p>
		</div>
<%
	  }
	}
}
%>
</div>
</body>
</html>

Servlet

messageServlet

package servlet;

import java.io.IOException;
import java.util.ArrayList;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import bean.Message;
import bean.User;
import dao.MessageDAO;
import dao.UserDAO;

public class MessageServlet extends HttpServlet{

	//インスタンス化
	MessageDAO messageDao = new MessageDAO();
	UserDAO userDao = new UserDAO();

	String error = "";
	String cmd = "";
    
    //メッセージ表示用
	private static final long serialVersionUID = 1L;
	public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

		// 文字エンコーディングの指定
		request.setCharacterEncoding("UTF-8");

		//パラメーターからデータ取得
		int roomid = Integer.parseInt(request.getParameter("roomid"));

		// セッション切れ例外処理(→logout)
		User user = (User) request.getSession().getAttribute("user");
		if (user == null) {
			error = "セッション切れの為、メッセージの投稿は出来ません。 ";
			cmd = "";
			return;
		}

		try {

			//メッセージリストを取得
			ArrayList<Message> messageList = messageDao.selectByroomid(roomid);

			//メッセージリストをリクエストスコープに登録
			request.setAttribute("messageList",messageList);
			//自身の情報を登録
			request.setAttribute("userinfo", userDao.selectByUserid(user.getUser_id()));
			//メッセージルームID登録
			request.setAttribute("roomid", roomid);

		} catch (IllegalStateException e) {
			error = "DB接続エラーの為、メッセージの表示は行えませんでした。";
			cmd = "menu";
			return;

		} finally {
			// 登録された件数を持ってフォワード
			if (error == ("")) {
				request.getRequestDispatcher("/view/messageroom.jsp").forward(request, response);

			} else {
				// 登録された件数を持ってフォワード
				request.setAttribute("error", error);
				request.setAttribute("cmd", cmd);
				request.getRequestDispatcher("/view/error.jsp").forward(request, response);
			}
		}

	}
    
    //メッセージ登録用
	public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

		// 文字エンコーディングの指定
		request.setCharacterEncoding("UTF-8");

		//パラメーターからルームID取得
		int roomid = Integer.parseInt(request.getParameter("roomid"));

		// セッション切れ例外処理(→logout)
		User user = (User) request.getSession().getAttribute("user");
		if (user == null) {
			error = "セッション切れの為、メッセージの投稿は出来ません。 ";
			cmd = "";
			return;
		}

		try {

			Message message =new Message();
			message.setMessage(request.getParameter("message"));
			message.setRoomid(roomid);
			message.setSenderId(user.getUser_id());
			messageDao.insert(message);
			//メッセージルームID登録
			request.setAttribute("roomid", roomid);

		} catch (IllegalStateException e) {
			error = "DB接続エラーの為、メッセージの投稿は行えませんでした。";
			cmd = "menu";
			return;

		} finally {
			// メッセージの登録を行い、画面へ遷移
			if (error == ("")) {
				request.getRequestDispatcher("/view/messageroom.jsp").forward(request, response);

			} else {
				// 登録された件数を持ってフォワード
				request.setAttribute("error", error);
				request.setAttribute("cmd", cmd);
				request.getRequestDispatcher("/view/error.jsp").forward(request, response);
			}
		}

	}
}

実際の動き

2022-07-18-00-53-29 (1).gif

参考文献

@OgawaNorihiroさんの下記の記事を参考にDBを作成しています。

最後に

これはあくまでも一例です。
研修時に作ったプログラムの機能を抜粋し、
本記事用に変換したものでこのままでは使うことが出来ない可能性があります。

3
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
1