基于WebSocket实现客户聊天室

目录

一、实现聊天室原理

二、聊天室前端代码

三、聊天室后端代码(重点)

四、聊天室实现效果展示


一、实现聊天室原理

1.1 介绍websocket协议

websocket是一种通信协议,再通过websocket实现弹幕聊天室时候,实现原理是客户端首先使用http协议请求服务器将通信协议转为websocket协议。

1.2、websocket的API

websocket分为客户端与服务器,其实现的API都不一样。

前端创建websocket案例:

<script>
let ws = new WebSocket("ws:/localhost/chat")
ws.open = function(){
};

ws.onmessage = function(evt){
    //通过evt.data 可以获取服务器发送的数据
};

ws.onclose = function(){
};
 
</script>

1.3 项目实现流程

1.4 项目结构


二、聊天室前端代码

前端核心在于两个:一个登陆界面,一个聊天室界面。

登陆界面:

<!DOCTYPE html>
<html lang="en">
<head>
    <title>聊天室-登录</title>
    <meta name="viewport" content="width=device-width, initial-scale=1"/>
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
    <meta name="keywords"
          content="Transparent Sign In Form Responsive Widget,Login form widgets, Sign up Web forms , Login signup Responsive web form,Flat Pricing table,Flat Drop downs,Registration Forms,News letter Forms,Elements"/>
    <script type="application/x-javascript">
        addEventListener("load", function () {
            setTimeout(hideURLbar, 0);
        }, false);

        function hideURLbar() {
            window.scrollTo(0, 1);
        }
    </script>

    <script src="js/jquery-1.9.1.min.js"></script>
    <link rel="icon" href="img/chat.ico" type="image/x-icon"/>
    <link rel="stylesheet" href="css/font-awesome.css"/> <!-- Font-Awesome-Icons-CSS -->
    <link rel="stylesheet" href="css/login.css" type="text/css" media="all"/> <!-- Style-CSS -->
</head>


<body class="background">
<div class="header-w3l">
    <h1>聊天室</h1>
</div>
<div class="main-content-agile" id="app">
    <div class="sub-main-w3">
        <h2>登录</h2>
        <form id="loginForm">
            <div class="icon1">
                <input placeholder="用户名" id="username" v-model="user.username" type="text"/>
            </div>

            <div class="icon2">
                <input placeholder="密码" id="password" v-model="user.password" type="password"/>
            </div>

            <div class="clear"></div>
            <input type="button" id="btn1" @click="login" value="登录"/>
			<div class="icon1">
                <span id="err_msg" style="color: red; ">{{errMessage}}</span>
            </div>
        </form>
    </div>
</div>
<div class="footer">
    <p>北京传智播客教育科技有限公司 版权所有Copyright 2006-2019  All Rights Reserved </p>
</div>
<script src="js/vue.js"></script>
<script src="js/axios-0.18.0.js"></script>
<script>
    new Vue({
        el:"#app",
        data() {
            return {
                errMessage: "",
                user:{
                    username:"",
                    password:""
                }
            }
        },
        methods: {
            login() {
                axios.post("user/login",this.user).then(res => {
                    //判断登陆是否成功
                    if(res.data.flag) {
                        location.href = "main.html"
                    } else {
                        this.errMessage = res.data.message;
                    }
                });
            }
        }
    });
</script>
</body>
</html>

效果:

聊天室页面:

<!DOCTYPE html>
<html lang="en">
<head>
    <title>黑马畅聊-登录</title>
    <meta name="viewport" content="width=device-width, initial-scale=1"/>
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
    <meta name="keywords"
          content="Transparent Sign In Form Responsive Widget,Login form widgets, Sign up Web forms , Login signup Responsive web form,Flat Pricing table,Flat Drop downs,Registration Forms,News letter Forms,Elements"/>
    <script type="application/x-javascript">
        addEventListener("load", function () {
            setTimeout(hideURLbar, 0);
        }, false);

        function hideURLbar() {
            window.scrollTo(0, 1);
        }
    </script>

    <script src="js/jquery-1.9.1.min.js"></script>
    <link rel="icon" href="img/chat.ico" type="image/x-icon"/>
    <link rel="stylesheet" href="css/font-awesome.css"/> <!-- Font-Awesome-Icons-CSS -->
    <link rel="stylesheet" href="css/login.css" type="text/css" media="all"/> <!-- Style-CSS -->
</head>


<body class="background">
<div class="header-w3l">
    <h1>聊天室</h1>
</div>
<div class="main-content-agile" id="app">
    <div class="sub-main-w3">
        <h2>登录</h2>
        <form id="loginForm">
            <div class="icon1">
                <input placeholder="用户名" id="username" v-model="user.username" type="text"/>
            </div>

            <div class="icon2">
                <input placeholder="密码" id="password" v-model="user.password" type="password"/>
            </div>

            <div class="clear"></div>
            <input type="button" id="btn1" @click="login" value="登录"/>
			<div class="icon1">
                <span id="err_msg" style="color: red; ">{{errMessage}}</span>
            </div>
        </form>
    </div>
</div>
<div class="footer">
    <p>北京传智播客教育科技有限公司 版权所有Copyright 2006-2019  All Rights Reserved </p>
</div>
<script src="js/vue.js"></script>
<script src="js/axios-0.18.0.js"></script>
<script>
    new Vue({
        el:"#app",
        data() {
            return {
                errMessage: "",
                user:{
                    username:"",
                    password:""
                }
            }
        },
        methods: {
            login() {
                axios.post("user/login",this.user).then(res => {
                    //判断登陆是否成功
                    if(res.data.flag) {
                        location.href = "main.html"
                    } else {
                        this.errMessage = res.data.message;
                    }
                });
            }
        }
    });
</script>
</body>
</html>


三、聊天室后端代码(重点)

3.1首先需要创建springboot项目,并导入以下jar包.

<!--        使用阿里巴巴的fastjson-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.78</version>
        </dependency>

<!--        实现websocket的jar包-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-websocket</artifactId>
        </dependency>

3.2 实现websocket,还需要对其进行配置,创建两个配置类

第一个进行websocketConfig的配置

@Configuration
public class WebsocketConfig {

// 将ServerEndPointExplorer加入ioc容器中
    @Bean
    public ServerEndpointExporter serverEndpointExporter() {
        return new ServerEndpointExporter();
    }
}

第二个配置是获取httpsession配置

public class GetHttpSessionConfig extends ServerEndpointConfig.Configurator {

    @Override
    public void modifyHandshake(ServerEndpointConfig sec, HandshakeRequest request, HandshakeResponse response) {
        //获取HttpSession对象
        HttpSession httpSession = (HttpSession) request.getHttpSession();
        //将httpSession对象保存起来
        sec.getUserProperties().put(HttpSession.class.getName(),httpSession);
    }
}

项目中需要ServerEndPoint存储所有用户的session对象,通过session实现用户之间的交流。所以还需要获取所有httpsession对象。

3.3 确定消息格式(JSON)

 为了确保实现的消息格式的准确,需要创建对应工具类,确保对应的格式的准确。

public class MessageUtils {

    public static String getMessage(boolean isSystemMessage,String fromName, Object message) {

        ResultMessage result = new ResultMessage();
        result.setSystem(isSystemMessage);
        result.setMessage(message);
        if(fromName != null) {
            result.setFromName(fromName);
        }
        return JSON.toJSONString(result);
    }
}

3.4 然后就是所有实体类的创建。

对于前端实体类,需要用户,封装http请求实体。

对于聊天室,需要用户发送的信息,服务器给用户发送的信息。

@Data
public class Result {
    private boolean flag;
    private String message;
}
@Data
public class User {

    private String userId;
    private String username;
    private String password;
}
@Data
public class Message {
    private String toName;
    private String message;
}
@Data
public class ResultMessage {

    private boolean isSystem;
    private String fromName;
    private Object message;//如果是系统消息是数组
}

3.5 后端基础功能的实现

后端实现登陆功能:用户名可以随意输入,但是密码必须是123。

    @PostMapping("/login")
    public Result login(@RequestBody User user, HttpSession session) {
        Result result = new Result();
        if(user != null && "123".equals(user.getPassword())) {
            result.setFlag(true);
            //将数据存储到session对象中
            session.setAttribute("user",user.getUsername());
        } else {
            result.setFlag(false);
            result.setMessage("登陆失败");
        }
        return result;
    }

后端实现获取用户名称功能:

 @GetMapping("/getUsername")
    public String getUsername(HttpSession session) {

        String username = (String) session.getAttribute("user");
        return username;
    }

 3.6 通过session发送消息的核心功能(重点)

@ServerEndpoint(value = "/chat",configurator = GetHttpSessionConfig.class)
@Component
public class ChatEndpoint {

    private static Map<String,Session> onlineUsers = new ConcurrentHashMap<>();

    static {
        // 初始化onlineUsers对象
        onlineUsers = new ConcurrentHashMap<>();
    }

    private HttpSession httpSession;

    /**
     * 建立websocket连接后,被调用
     * @param session
     */
    @OnOpen
    public void onOpen(Session session, EndpointConfig config) {
        //1,将session进行保存
        this.httpSession = (HttpSession) config.getUserProperties().get(HttpSession.class.getName());
        String user = (String) this.httpSession.getAttribute("user");
        onlineUsers.put(user,session);
        //2,广播消息。需要将登陆的所有的用户推送给所有的用户
        String message = MessageUtils.getMessage(true,null,getFriends());
        broadcastAllUsers(message);
    }

    public Set getFriends() {
        Set<String> set = onlineUsers.keySet();
        return set;
    }

    private void broadcastAllUsers(String message) {
        try {
            //遍历map集合
            Set<Map.Entry<String, Session>> entries = onlineUsers.entrySet();
            for (Map.Entry<String, Session> entry : entries) {
                //获取到所有用户对应的session对象
                Session session = entry.getValue();
                //发送消息
                session.getBasicRemote().sendText(message);
            }
        } catch (Exception e) {
            //记录日志
        }
    }

    /**
     * 浏览器发送消息到服务端,该方法被调用
     *
     * 张三  -->  李四
     * @param message
     */
    @OnMessage
    public void onMessage(String message) {
        try {
            //将消息推送给指定的用户
            Message msg = JSON.parseObject(message, Message.class);
            //获取 消息接收方的用户名
            String toName = msg.getToName();
            String mess = msg.getMessage();
            //获取消息接收方用户对象的session对象
            Session session = onlineUsers.get(toName);
            String user = (String) this.httpSession.getAttribute("user");
            String msg1 = MessageUtils.getMessage(false, user, mess);
            session.getBasicRemote().sendText(msg1);
        } catch (Exception e) {
            //记录日志
        }
    }

    /**
     * 断开 websocket 连接时被调用
     * @param session
     */
    @OnClose
    public void onClose(Session session) {
        //1,从onlineUsers中剔除当前用户的session对象
        String user = (String) this.httpSession.getAttribute("user");
        onlineUsers.remove(user);
        //2,通知其他所有的用户,当前用户下线了
        String message = MessageUtils.getMessage(true,null,getFriends());
        broadcastAllUsers(message);
    }
}

代码重点在于对session的使用,将内容放在对应用户session的共享域中,后端则负责统一管理所有用户的session。


四、聊天室实现效果展示

4.1 登陆功能:

4.2 在线与不在线,当后端运行后才能显示在线

4.3 当有其他人登陆时候弹出对应用户名称

4.4 通过点击对应的名称进行一对一聊天

4.5 实现聊天功能


 

  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: Websocket聊天室是一种基于Websocket技术的实时通讯应用,可以实现多人在线聊天、发送图片、文件等功能。它是一种全双工通信协议,通过建立客户端和服务器之间的长连接,实现双向通信。相比于传统的HTTP协议,Websocket协议具有更低的延迟、更高的效率、更少的网络负荷等优点,因此被广泛应用于实时通讯领域。 Websocket聊天室实现可以分为客户端和服务器两个部分。客户端一般是通过浏览器提供的Websocket API来实现,而服务端则需要使用专门的Websocket服务器,例如Node.js的Socket.IO、Java的Tomcat等。客户端和服务器之间通过Websocket协议进行通信,客户端可以发送消息、文件等数据,服务器则接收消息并进行处理,然后将消息转发给其他客户端。 Websocket聊天室实现过程中需要考虑多个方面,例如安全性、可扩展性、性能等问题。例如,需要对用户身份进行认证、对数据进行加密传输、对消息进行过滤和校验等。同时,需要考虑如何处理大量的并发连接,如何保证消息的可靠性、顺序性等问题。 总之,Websocket聊天室是一种非常实用的实时通讯应用,可以为用户提供高效、便捷、可靠的聊天体验。 ### 回答2: WebSocket是一种用于在浏览器和服务器之间进行实时双向通信的通信协议。WebSocket聊天室是一个基于WebSocket技术的实时聊天应用。 WebSocket聊天室通过WebSocket协议建立了一个双向通信通道,在浏览器和服务器之间实现实时的消息传递。用户可以在浏览器中发送消息,服务器会将消息实时地推送给其他在线用户,其他用户也可以实时地向服务器发送消息,服务器会将这些消息分发给其他在线用户。 WebSocket聊天室具有以下特点: 1. 实时性:WebSocket聊天室实现了服务器和浏览器之间的实时双向通信,用户可以实时地发送和接收消息。 2. 高效性:与传统的轮询方式相比,WebSocket采用了事件驱动的方式,减少了不必要的数据传输,提高了通信效率。 3. 可扩展性:WebSocket聊天室可以添加更多功能,如发送图片、语音等,增加用户体验。 4. 安全性:WebSocket聊天室可以通过使用SSL/TLS协议进行加密,确保通信过程的安全性和用户信息的保密性。 5. 跨平台:WebSocket聊天室支持多种浏览器和操作系统,用户可以在不同的设备上使用。 总之,WebSocket聊天室是一种基于WebSocket协议的实时聊天应用,通过实现浏览器和服务器之间的双向通信,实现用户之间的实时消息传递。它具有实时性、高效性、可扩展性、安全性和跨平台等特点,是一种很好的实时通信解决方案。 ### 回答3: WebSocket聊天室是一种基于WebSocket协议实现的实时通信的应用程序。与传统的HTTP请求-响应模式不同,WebSocket提供了一个持久的双向通信通道,允许服务器主动向客户端推送数据。 在WebSocket聊天室中,用户可以通过浏览器或类似应用程序连接到服务器,建立起WebSocket连接。一旦连接建立成功,用户可以与其他连接到同一服务器的人实时交流,发送和接收消息。 聊天室通常包括以下功能: 1. 用户认证:用户可以通过提供用户名和密码等凭证进行认证,确保只有授权用户才能进入聊天室。 2. 房间管理:聊天室可以分为多个房间,用户可以选择进入感兴趣的房间。聊天室管理员可以创建、编辑和删除房间。 3. 消息发送:用户可以通过输入框输入消息并发送给聊天室中的其他人。发送的消息可以是文本、图片、文件等。 4. 消息接收:当有新消息时,聊天室会将该消息实时推送给所有连接到聊天室的用户,用户可以看到其他人发送的消息。 5. 在线用户列表:聊天室可以显示当前在线的用户列表,包括用户名、头像等信息。用户可以通过点击在线用户的头像或用户名与之私聊。 6. 表情和文件上传:聊天室可以支持发送表情和文件。用户可以选择从本地上传文件并发送给聊天室中的其他人。 WebSocket聊天室可以用于各种场景,如在线客服、团队协作、社交网络等。它提供了实时通信的能力,可以有效地促进人与人之间的交流和沟通。同时,它也需要服务器端提供相应的支持和处理逻辑,保证聊天室的稳定运行和安全性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Alphamilk

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值