Netty基础,Netty实现群聊系统

article/2025/10/23 5:31:35

NIO群聊系统

这里面的知识比较全面,用到了我们之前学习的三大组件,首先我先来给大家介绍本系统的功能

服务端功能

最基本的当然是注册功能,也就是将serverSocketChannel注册进Selector,Selector负责调度事件

监听、读取客户端发来的消息

收到客户端的信息后将其转发到其它客户端,完成群聊功能

客户端功能

连接服务端,注册进Selector

读写消息

以上就是群聊系统最基本的功能,我们直接上代码,每句话的含义我都标在它的顶上了

服务端源码

package com.hecl.designmodels.NIO.GroupChat;import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.*;
import java.util.Iterator;
import java.util.Set;public class GroupChatServer {//定义一个选择器对象private Selector selector;//定义一个ServerSocketChannel对象private ServerSocketChannel serverSocketChannel;//养成习惯,对于这种不变的属性我们使用static final 来定义,且单词为全大写private static final int PORT = 7000;public GroupChatServer() {try{//获取选择器selector = Selector.open();//获取ServerSocketChannelserverSocketChannel = ServerSocketChannel.open();//绑定端口serverSocketChannel.socket().bind(new InetSocketAddress(PORT));//设置为非阻塞serverSocketChannel.configureBlocking(false);//将serverSocketChannel注册进selectorserverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);}catch (Exception e){e.printStackTrace();}}//监听方法public void listen(){try{while(true){int count = selector.select();//如果count大于0就说明有事件发生if(count > 0){//selector.selectedKeys()获取一个Set<SelectionKey>集合//遍历SelectionKeyIterator<SelectionKey> iterator =  selector.selectedKeys().iterator();while (iterator.hasNext()){//获取目前遍历到的SelectionKey对象SelectionKey key = iterator.next();//有新连接创建if(key.isAcceptable()){//获取连接到服务器的SocketChannelSocketChannel socketChannel = serverSocketChannel.accept();//将该通道设置为非阻塞socketChannel.configureBlocking(false);//注册进selectorsocketChannel.register(selector,SelectionKey.OP_READ);System.out.println(socketChannel.getRemoteAddress() + ": 已登录");}//发生有数据发送过来触发读事件if(key.isReadable()){//调用读取数据的方法readMsg(key);}iterator.remove();}}}}catch (IOException e){e.printStackTrace();}}//读取数据方法public void readMsg(SelectionKey selectionKey){SocketChannel socketChannel = null;try {//通过key获取与之关联的SocketChannelsocketChannel = (SocketChannel) selectionKey.channel();//创建缓冲区去接收通道内的数据ByteBuffer byteBuffer = ByteBuffer.allocate(2048);//得到通道内的数据socketChannel.read(byteBuffer);//在控制台打印消息System.out.println("from " + socketChannel.getRemoteAddress() + "客户端 :" + new String(byteBuffer.array()));//转发消息sendMsgToOtherClients(new String(byteBuffer.array()),socketChannel);}catch (IOException e){try{System.out.println(socketChannel.getRemoteAddress() + "下线");//消毁selectionKeyselectionKey.cancel();//销毁socketChannelsocketChannel.close();}catch (Exception exception){exception.printStackTrace();}}}//向其他客户端转发消息,不包括自己public void sendMsgToOtherClients(String msg,SocketChannel socketChannel){try {//获取所有的SelectionKeyfor(SelectionKey OtherKey : selector.keys()){//获取对应的socketChannelChannel OtherChannel = OtherKey.channel();//排除自身if(OtherChannel instanceof SocketChannel && socketChannel != OtherChannel){//向下转型SocketChannel OtherSocketChannel = (SocketChannel) OtherChannel;//创建缓冲区ByteBuffer byteBuffer = ByteBuffer.wrap(msg.getBytes());//将缓冲区的数据写入通道OtherSocketChannel.write(byteBuffer);}}}catch (IOException e){e.printStackTrace();}}public static void main(String[] args) {//启动服务端GroupChatServer groupChatServer = new GroupChatServer();//调用监听方法groupChatServer.listen();}
}

客户端源码

package com.hecl.designmodels.NIO.GroupChat;import com.sun.xml.internal.ws.policy.privateutil.PolicyUtils;
import designModels.design_mode_07_BridgePattern.Pay;import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;
import java.util.Iterator;
import java.util.Scanner;
import java.util.Set;public class GroupChatClient {//定义需要连接服务器的地址与端口号private static final String HOST = "127.0.0.1";private static final int PORT = 7000;//定义选择器对象private Selector selector;//定义SocketChannel对象private SocketChannel socketChannel;//登录名private String username;public GroupChatClient(String username){try {//获取一个SocketChannel对象,并连接服务器socketChannel = SocketChannel.open(new InetSocketAddress(HOST,PORT));//得到选择器selector = Selector.open();//设置非阻塞socketChannel.configureBlocking(false);//将通道注册进selectorsocketChannel.register(selector, SelectionKey.OP_READ);//得到登录名称this.username = username;String s = username + " 加入群聊";System.out.println(s);socketChannel.write(ByteBuffer.wrap(s.getBytes()));}catch (IOException e){e.printStackTrace();}}public void sendMsg(String msg){//数据拼接形成最终发送的消息String info = username + " say : " + msg;try{//将缓冲区内的数据写入通道socketChannel.write(ByteBuffer.wrap(info.getBytes()));System.out.println("I say : " + msg);}catch (IOException e){e.printStackTrace();}}//参照服务端读取数据代码public void readMsg(){try{int count = selector.select();if(count>0){Set<SelectionKey> selectionKeySet = selector.selectedKeys();Iterator<SelectionKey> iterator = selectionKeySet.iterator();while (iterator.hasNext()){SelectionKey selectionKey = iterator.next();if(selectionKey.isReadable()){SocketChannel socketChannel = (SocketChannel) selectionKey.channel();ByteBuffer byteBuffer = ByteBuffer.allocate(2048);socketChannel.read(byteBuffer);String msg = new String(byteBuffer.array());System.out.println(msg);}//必须要移除,要不然收不到其它客户端的消息iterator.remove();}}}catch (IOException e){e.printStackTrace();}}public static void main(String[] args) {Scanner scannerName = new Scanner(System.in);//启动服务GroupChatClient groupChatClient = new GroupChatClient(scannerName.nextLine());//新建一个线程循环调用读取数据的方法new Thread() {public void run(){while(true) {groupChatClient.readMsg();try {//休息2sThread.sleep(2000);}catch (Exception e){e.printStackTrace();}}}}.start();//获取输入消息Scanner scanner = new Scanner(System.in);while (scanner.hasNextLine()){//获取输入消息并调用sendMsg(String msg)groupChatClient.sendMsg(scanner.nextLine());}}}

我们来看看最终实现效果

服务器端
在这里插入图片描述
客户端1
在这里插入图片描述
客户端2
在这里插入图片描述
客户端3
在这里插入图片描述
在这里插入图片描述
前面发送的消息,后加入的客户端获取不到,这就是通过NIO实现的简单群聊啦

基础都打得差不多了,下一篇我们就应该正式进入Netty的章节啦,赞赞赞赞赞

完成:2021/3/28 20:20 ALiangX

转载请标注原作者,谢谢你们的支持,能给个小心心吗?
在这里插入图片描述


http://chatgpt.dhexx.cn/article/t85HHd9g.shtml

相关文章

c语言 多进程实现基于UDP的网络群聊聊天室

功能 有新用户登录&#xff0c;其他在线的用户可以收到登录信息 有用户群聊&#xff0c;其他在线的用户可以收到群聊信息 有用户退出&#xff0c;其他在线的用户可以收到退出信息 服务器可以发送系统信息 流程图如下&#xff1a; 提示&#xff1a; 客户端登录之后&#x…

Spring Boot使用WebSocket实现群聊

1.通过https://start.aliyun.com创建一个spring boot项目 https://start.aliyun.com 所需依赖&#xff1a; 2.编写代码 目录&#xff1a; WebMvcConfig&#xff0c;监控用户页面&#xff1a; package com.example.mywbsk.config;import org.springframework.context.annotat…

网络编程 : 基于UDP的网络群聊聊天室

一、UDP网络编程: 1.1 流程 服务器流程: 创建用户数据报套接字 填充服务器的网络信息结构体 绑定套接字与服务器网络信息结构体 收发数据 关闭套接字 客户端流程&#xff1a; 创建用户数据报套接字 填充服务器的网络信息结构体 收发数据 关闭套接字 二、基于UDP的网络群聊聊天…

Java网络编程:TCP实现群聊功能代码

Java网络编程&#xff1a;TCP实现群聊功能代码 实现的功能&#xff1a;群聊 具体操作流程的大致思路&#xff1a; 某个客户端发送消息&#xff0c;服务器将该消息转发给其余客户端&#xff08;除了自己&#xff09;。 代码一共分两个端&#xff08;服务器和客户端&#xff0…

【Java教程】UDP实现群聊聊天室

大家好&#xff0c;今天为大家带来了一个非常有意思的小程序——UDP实现的群聊聊天室。这个程序使用的UDP协议&#xff0c;并使用DatagramSocket的子类MulticastSocket实现组播&#xff0c;可以部署在一个局域网内的多台电脑上&#xff0c;并可以实现文字群聊。 本文将会按照以…

群聊私聊天建群社交即时通讯H5系统开发

群聊私聊天建群社交即时通讯H5系统开发 前端功能&#xff1a; 聊天、通讯录、动态、发现、我、多国语言、私聊、群聊、创建群聊&#xff08;设置免费、收费&#xff09;、发布动态、发信息&#xff08;图片、文字、红包、不支持语言&#xff09;、我的余额&#xff08;后台添加…

socket.io实现简易版群聊

最近学了点 websocket 的技术知识&#xff0c;了解到 node.js 有相关技术——socket.io可以实现&#xff0c;就想着实现一个简单版本的群聊。 先看看效果图 在浏览器中打开2个窗口&#xff0c;访问 http://localhost:3000/socket 目录结构 node 服务器代码实现 需要安装 exp…

Go实现简易聊天室(群聊)

参考&#xff1a;Go 群聊 ( goroutine ) 语雀 基于websocket的聊天室&#xff0c;可进一步参考&#xff1a; (1) go实现聊天室(WebSocket方式) (2) Golang代码搜集-基于websocketvue.js的简易聊天室 闲着无聊ing~ 一直想着能用go整一些好玩的小工具&#xff0c;想着想着突然想…

vue+websocket+nodejs创建聊天室- 创建群聊、加入群聊

前言 前几篇我们讲了如何创建聊天室,一对一/一对多聊天&#xff0c;以及加已读未读消息状态。 这篇主要讲如何创建群聊和加入群聊。 还是跟之前一样&#xff0c;本文只写新加入的逻辑&#xff0c;之前逻辑请查看之前文章。 PS&#xff1a;效果稍微有点粗糙哈哈&#xff0c;不要…

html群聊插件,团队群聊.html

&#xfeff;团队群聊 $axure.utils.getTransparentGifPath function() { return resources/images/transparent.gif; }; $axure.utils.getOtherPath function() { return resources/Other.html; }; $axure.utils.getReloadPath function() { return resources/reload.html;…

ajax聊天室创建群聊,js+node.js+socket.io实现聊天功能(私聊,创建群聊)

效果图: 这里启动了四个客户端进行测试 1. 登录,以及获取在线用户列表 2. 私聊功能 3. 群聊功能 偶然发现了WebSocket, 发现这个可以实时通信,在线聊天,所以就做了一个聊天工具的demo,记录一下 Socket.io WebSocket是js原生自带的,而Socket.io相当于是对WebSocket进行封装…

php怎么做群聊,workerman实现群聊

服务端&#xff1a; require_once "../vendor/autoload.php"; use Workerman\Worker; $worker new Worker(‘websocket://0.0.0.0:2345‘); $worker->count 1; $worker->onWorkerStart function($data){ echo "连接开始\n"; }; $worker->onCon…

致微信:我们什么时候重新定义群聊?

你永远都不知道&#xff0c;自己到底有多少微信群。 你也很难想象&#xff0c;一个20多人的小团队&#xff0c;相互之间会建几百个群&#xff0c;然后当你需要群发一个文件/通知时&#xff0c;还是找不到合适的群&#xff0c;怎么办&#xff1f;重新拉一个群呗。 如果你加的群稍…

怎么玩转LINE 群聊

关键词&#xff1a;LINE群聊 &#xff1b;SaleSmartly&#xff08;ss客服&#xff09; 一个 LINE 群聊是企业让客户了解其产品或服务的有效方式。如果使用得当&#xff0c; LINE 小组是企业同时快速培养与多个客户关系的绝佳场所。想了解更多吗&#xff1f;继续阅读&#xff0…

微信怎么找群聊?找回微信群聊只需要这样…

如今社会中的日常生活已经离不开微信&#xff0c;微信是社交界的一股清流&#xff0c;聊天、视频、语音等等功能&#xff0c;而且支付功能的用途范围在国内也甚是广泛。微信怎么找群聊&#xff1f;是的&#xff0c;当多个好友之间相互认识&#xff0c;或者都有一个共同点的时候…

免费tk域名+freewebhostingarea空间

1.申请免费域名 进入http://www.dot.tk&#xff08;推荐注册tk域名&#xff09;&#xff0c;申请一个新的域名&#xff0c;每次申请12个月以下是免费的&#xff0c;到期前14天可以免费续期 在此页面执行下一步之前&#xff0c;需要进行设置DNS服务器 2.获取DNS服务器 申请进…

免费国外PHP+MYSQL空间申请

这是一个在匈牙利的免费空间,但是在网页上方挂广告条(可以用CSS屏蔽掉). 支持的以下内容管理系统: - Joomla - Wordpress - e107 - Drupal - Simple Machine Forum (SMF) - PHP-Nuke - phpBB - Elxis - MyBB - CMS Made Simple 具体申请步骤: 1.进入www.freeweb.hu 2. 3. 4. 5.…

GIS地理空间数据免费获取

GIS地理空间数据免费获取 国内&#xff1a; 一、测绘地理信息局会提供权威的数据。 需要进入全国地理信息资源目录服务系统网站&#xff08;http://www.webmap.cn/main.do?methodindex&#xff09;&#xff0c;该网站提供&#xff1a;30米全球地表覆盖数据&#xff0c;Globe…

15 个国外免费卫星图像数据源介绍

有人说&#xff1a;一个人从1岁活到80岁很平凡&#xff0c;但如果从80岁倒着活&#xff0c;那么一半以上的人都可能不凡。 生活没有捷径&#xff0c;我们踩过的坑都成为了生活的经验&#xff0c;这些经验越早知道&#xff0c;你要走的弯路就会越少。 数据源 (Data Source) 顾名…

新手建站十大免费空间推荐-稳定,可用的免费空间及其使用体验

新手建站十大免费空间推荐-稳定,可用的免费空间及其使用体验 新手建站十大 免费空间推荐-稳定,可用的 免费空间及其使用体验 一、OpenShift OpenShift 1、OpenShift空间是Red Hat的平台即服务的云计算平台(PaaS)&#xff0c;自成立以来就为大家提供 免费空间服务&#xff0c;深…