Netty案例介绍-群聊案例实现
【摘要】
前面介绍了基于Netty的Http服务,本文我们来通过Netty来实现群聊案例。
群聊案例
1.案例需求
编写一个 Netty 群聊系统,实现服务器端和客户端之间的数据简单通讯(非阻塞)实现多人...
前面介绍了基于Netty的Http服务,本文我们来通过Netty来实现群聊案例。
群聊案例
1.案例需求
- 编写一个 Netty 群聊系统,实现服务器端和客户端之间的数据简单通讯(非阻塞)
- 实现多人群聊
- 服务器端:可以监测用户上线,离线,并实现消息转发功能
- 客户端:通过channel 可以无阻塞发送消息给其它所有用户,同时可以接受其它用户发送的消息(有服务器转发得到)
目的:进一步理解Netty非阻塞网络编程机制
2.服务端代码
2.1 服务端处理器
在服务端处理器中我们要处理客户端的上下线及消息的分发
package com.dpb.netty.goupchat;
import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.channel.group.ChannelGroup;
import io.netty.channel.group.DefaultChannelGroup;
import io.netty.util.concurrent.GlobalEventExecutor;
import java.text.SimpleDateFormat;
import java.util.Date;
/**
* @program: netty4demo
* @description: 群聊服务处理器
* 1.客户端连接 提示其他客户端上线
* 2.客户端发送消息,需要将详细转发给其他客户端
* @author: 波波烤鸭
* @create: 2019-12-29 14:02
*/
public class GroupChatServerHandler extends SimpleChannelInboundHandler<String> {
// 管理所有客户端的Channel 是一个单例对象
private static final ChannelGroup grouup = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE);
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH-mm-ss");
/**
* 客户端连接触发的方法
* @param ctx
* @throws Exception
*/
@Override
public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
// 获取当前连接对象的 Channel对象
Channel channel = ctx.channel();
// 通知其他客户端 当前客户端连接上线了
grouup.writeAndFlush("[客户端上线了]" + channel.remoteAddress() + " 上线了 " + sdf.format(new Date()) + "\n");
// 将当前channel添加到 channelGroup中
grouup.add(channel);
}
/**
* 客户端断开连接触发的方法
* @param ctx
* @throws Exception
*/
@Override
public void handlerRemoved(ChannelHandlerContext ctx) throws Exception {
// 通知其他客户端 我离线了
ctx.writeAndFlush("[客户端下线了]" + ctx.channel().remoteAddress() + " : " + sdf.format(new Date())+ "\n");
}
/**
* 读取客户端请求消息的方法
* @param context
* @param s
* @throws Exception
*/
@Override
protected void channelRead0(ChannelHandlerContext context, String s) throws Exception {
// 获取当前的 handler
Channel channel = context.channel();
// 遍历channelGroup
grouup.forEach(ch ->{
if(ch == channel){
// 是自己
ch.writeAndFlush( "我说:" + s);
}else{
// 不是自己
ch.writeAndFlush(channel.remoteAddress() + "说:" + s);
}
});
}
}
- 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
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
- 76
- 77
- 78
- 79
2.2 服务端
服务端我们要启动我们的服务,绑定端口等。
package com.dpb.netty.goupchat;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.codec.string.StringEncoder;
/**
* @program: netty4demo
* @description: 群聊系统服务端
* @author: 波波烤鸭
* @create: 2019-12-29 13:50
*/
public class GroupChatServer {
// 服务的端口号
private int port;
public GroupChatServer(int port){
this.port = port;
}
/**
* 服务运行的方法
*/
public void run() throws Exception{
// 创建BossGroup和WorkGroup
EventLoopGroup bossGroup = new NioEventLoopGroup(1);
EventLoopGroup workGroup = new NioEventLoopGroup();
// 创建 ServerBootstrap 服务启动对象
ServerBootstrap bootstrap = new ServerBootstrap();
try{
// 给bootstrap设置相关的参数
bootstrap.group(bossGroup,workGroup)
.channel(NioServerSocketChannel.class)
.option(ChannelOption.SO_BACKLOG,128)
.childOption(ChannelOption.SO_KEEPALIVE,true)
.childHandler(new ChannelInitializer<SocketChannel>() {
@Override
protected void initChannel(SocketChannel sc) throws Exception {
// 获取pipeline对象
ChannelPipeline pipeline = sc.pipeline();
// 设置对应的handler
pipeline.addLast("docoder",new StringDecoder());
pipeline.addLast("encoder",new StringEncoder());
pipeline.addLast(new GroupChatServerHandler());
}
});
System.out.println("服务端启动了.......");
// 绑定端口
ChannelFuture future = bootstrap.bind(port).sync();
future.channel().closeFuture().sync();
}finally {
bossGroup.shutdownGracefully();
workGroup.shutdownGracefully();
}
}
public static void main(String[] args) throws Exception{
GroupChatServer server = new GroupChatServer(8888);
server.run();
}
}
- 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
- 60
- 61
- 62
- 63
- 64
- 65
- 66
3. 客户端代码
3.1 客户端处理器
获取服务器转发的消息
package com.dpb.netty.goupchat;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
/**
* @program: netty4demo
* @description: 客户端处理器
* @author: 波波烤鸭
* @create: 2019-12-29 14:19
*/
public class GroupChatClientHandler extends SimpleChannelInboundHandler<String> {
/**
* 读取消息
* @param channelHandlerContext
* @param s
* @throws Exception
*/
@Override
protected void channelRead0(ChannelHandlerContext channelHandlerContext, String s) throws Exception {
System.out.println(s.trim());
}
}
- 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
3.2 客户端代码
连接服务器,发送消息
package com.dpb.netty.goupchat;
import io.netty.bootstrap.Bootstrap;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.codec.string.StringEncoder;
import java.util.Scanner;
/**
* @program: netty4demo
* @description:
* @author: 波波烤鸭
* @create: 2019-12-29 14:14
*/
public class GroupChatClient {
public String host;
public int port ;
public GroupChatClient(String host,int port){
this.host = host;
this.port = port;
}
public void run() throws Exception{
EventLoopGroup clientGroup = new NioEventLoopGroup(1);
Bootstrap bootstrap = new Bootstrap();
try{
bootstrap.group(clientGroup)
.channel(NioSocketChannel.class)
.handler(new ChannelInitializer<SocketChannel>() {
@Override
protected void initChannel(SocketChannel sc) throws Exception {
sc.pipeline().addLast("decoder",new StringDecoder());
sc.pipeline().addLast("encoder",new StringEncoder());
sc.pipeline().addLast(new GroupChatClientHandler());
}
});
ChannelFuture future = bootstrap.connect(host, port).sync();
Channel channel = future.channel();
// 发送消息
Scanner scanner = new Scanner(System.in);
while(scanner.hasNextLine()){
String msg = scanner.nextLine();
channel.writeAndFlush(msg + "\n");
}
}finally {
clientGroup.shutdownGracefully();
}
}
public static void main(String[] args) throws Exception {
GroupChatClient client = new GroupChatClient("localhost",8888);
client.run();
}
}
- 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
- 60
- 61
- 62
- 63
- 64
- 65
4.测试
启动服务器,然后启动多个客户端查看效果
查看效果,功能实现~
文章来源: dpb-bobokaoya-sm.blog.csdn.net,作者:波波烤鸭,版权归原作者所有,如需转载,请联系作者。
原文链接:dpb-bobokaoya-sm.blog.csdn.net/article/details/103754088
【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱:
cloudbbs@huaweicloud.com
- 点赞
- 收藏
- 关注作者
评论(0)