基于Netty5.0入门案例六之NettyServer群发消息
前言介绍:
我们的NettyServer收到数据后,需要群发给当前链接到服务端的所有小伙伴。
技术点:
1、ChannelGroup 【io.netty.channel.group.DefaultChannelGroup】
欢迎加入:itstack | Netty The Sniper 5360692
环境需求:
1、jdk1.7以上【jdk1.7以下只能部分支持netty】
2、Netty-all-5.0【netty3.x 4.x 5每次的变化较大,接口类名也随着变化】
3、telnet 测试【可以现在你的win7机器上测试这个命令,用于链接到服务端的测试命令】【本案例中已经很不好满足测试需求了】
4、最好下载个网络调试助手,它能帮助你测试服务端、客户端
代码部分:
======================
TestNettyServerBaseDemo
src
com.itstack
ChildChannelHandler.java
MyChannelHandlerPool.java
MyServerHanlder.java
NettyServer.java
======================
ChildChannelHandler.java
- package com.itstack;
- import io.netty.buffer.ByteBuf;
- import io.netty.buffer.Unpooled;
- import io.netty.channel.ChannelInitializer;
- import io.netty.channel.socket.SocketChannel;
- import io.netty.handler.codec.DelimiterBasedFrameDecoder;
- import io.netty.handler.codec.Delimiters;
- import io.netty.handler.codec.FixedLengthFrameDecoder;
- import io.netty.handler.codec.LineBasedFrameDecoder;
- import io.netty.handler.codec.string.StringDecoder;
- import io.netty.handler.codec.string.StringEncoder;
- public class ChildChannelHandler extends ChannelInitializer<SocketChannel> {
- @Override
- protected void initChannel(SocketChannel e) throws Exception {
- System.out.println("报告");
- System.out.println("信息:有一客户端链接到本服务端");
- System.out.println("IP:" + e.localAddress().getHostName());
- System.out.println("Port:" + e.localAddress().getPort());
- System.out.println("报告完毕");
- // 解码器
- // 基于换行符号
- e.pipeline().addLast(new LineBasedFrameDecoder(1024));
- // 基于指定字符串【换行符,这样功能等同于LineBasedFrameDecoder】
- // e.pipeline().addLast(new DelimiterBasedFrameDecoder(1024, false, Delimiters.lineDelimiter()));
- // 基于最大长度
- // e.pipeline().addLast(new FixedLengthFrameDecoder(4));
- // 解码转String
- e.pipeline().addLast(new StringDecoder());
- // 编码器 String
- e.pipeline().addLast(new StringEncoder());
- // 在管道中添加我们自己的接收数据实现方法
- e.pipeline().addLast(new MyServerHanlder());
- }
- }
MyChannelHandlerPool.java
- package com.itstack;
- import io.netty.channel.group.ChannelGroup;
- import io.netty.channel.group.DefaultChannelGroup;
- import io.netty.util.concurrent.GlobalEventExecutor;
- /**
- *
- * 这里讲ChannelGroup单独放到一个类里,并有多个客户端使用
- * 同时ChannelGroup是static的
- * 说明:这不是唯一的处理方式
- *
- */
- public class MyChannelHandlerPool {
- public static ChannelGroup channelGroup = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE);
- }
MyServerHanlder.java
- package com.itstack;
- import java.util.Date;
- import io.netty.buffer.ByteBuf;
- import io.netty.buffer.Unpooled;
- import io.netty.channel.ChannelHandlerAdapter;
- import io.netty.channel.ChannelHandlerContext;
- import io.netty.handler.codec.bytes.ByteArrayDecoder;
- public class MyServerHanlder extends ChannelHandlerAdapter{
- /*
- * channelAction
- *
- * channel 通道
- * action 活跃的
- *
- * 当客户端主动链接服务端的链接后,这个通道就是活跃的了。也就是客户端与服务端建立了通信通道并且可以传输数据
- *
- */
- public void channelActive(ChannelHandlerContext ctx) throws Exception {
- System.out.println(ctx.channel().localAddress().toString()+" channelActive");
- //添加到channelGroup 通道组
- MyChannelHandlerPool.channelGroup.add(ctx.channel());
- //通知您已经链接上客户端
- String str = "您已经开启与服务端链接"+" "+ctx.channel().id()+new Date()+" "+ctx.channel().localAddress();
- ctx.writeAndFlush(str);
- }
- /*
- * channelInactive
- *
- * channel 通道
- * Inactive 不活跃的
- *
- * 当客户端主动断开服务端的链接后,这个通道就是不活跃的。也就是说客户端与服务端的关闭了通信通道并且不可以传输数据
- *
- */
- public void channelInactive(ChannelHandlerContext ctx) throws Exception {
- // 从channelGroup中移除,当有客户端退出后,移除channel。
- MyChannelHandlerPool.channelGroup.remove(ctx.channel());
- System.out.println(ctx.channel().localAddress().toString()+" channelInactive");
- }
- /*
- * channelRead
- *
- * channel 通道
- * Read 读
- *
- * 简而言之就是从通道中读取数据,也就是服务端接收客户端发来的数据
- * 但是这个数据在不进行解码时它是ByteBuf类型的后面例子我们在介绍
- *
- */
- public void channelRead(ChannelHandlerContext ctx, Object msg)
- throws Exception {
- //注意此处已经不需要手工解码了
- System.out.println(ctx.channel().id()+""+new Date()+" "+msg);
- //通知您已经链接上客户端[给客户端穿回去的数据加个换行]
- String str = "服务端收到:"+ctx.channel().id()+new Date()+" "+msg+"\r\n";
- //收到信息后,群发给所有小伙伴
- MyChannelHandlerPool.channelGroup.writeAndFlush(str);
- }
- /*
- * channelReadComplete
- *
- * channel 通道
- * Read 读取
- * Complete 完成
- *
- * 在通道读取完成后会在这个方法里通知,对应可以做刷新操作
- * ctx.flush()
- *
- */
- public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
- ctx.flush();
- }
- /*
- * exceptionCaught
- *
- * exception 异常
- * Caught 抓住
- *
- * 抓住异常,当发生异常的时候,可以做一些相应的处理,比如打印日志、关闭链接
- *
- */
- public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause)
- throws Exception {
- ctx.close();
- System.out.println("异常信息:\r\n"+cause.getMessage());
- }
- }
NettyServer.java
- package com.itstack;
- import io.netty.bootstrap.ServerBootstrap;
- import io.netty.channel.ChannelFuture;
- import io.netty.channel.ChannelOption;
- import io.netty.channel.EventLoopGroup;
- import io.netty.channel.nio.NioEventLoopGroup;
- import io.netty.channel.socket.nio.NioServerSocketChannel;
- public class NettyServer {
- public static void main(String[] args) {
- try {
- System.out.println("服务端开启等待客户端链接");
- new NettyServer().bing(7397);
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- public void bing(int port) throws Exception{
- EventLoopGroup bossGroup = new NioEventLoopGroup();
- EventLoopGroup workGroup = new NioEventLoopGroup();
- try {
- ServerBootstrap b = new ServerBootstrap();
- b.group(bossGroup, workGroup);
- b.channel(NioServerSocketChannel.class);
- b.option(ChannelOption.SO_BACKLOG, 1024);
- b.childHandler(new ChildChannelHandler());
- // 绑定端口
- ChannelFuture f = b.bind(port).sync();
- // 等待服务端监听端口关闭
- f.channel().closeFuture().sync();
- } finally {
- // 优雅的退出
- bossGroup.shutdownGracefully();
- workGroup.shutdownGracefully();
- }
- }
- }
1、启动NettyServer
2、控制台输出:
----------------------------------------------
服务端开启等待客户端链接
----------------------------------------------
3、开启2个以上客户端模拟软件
4、
服务端端控制台输出:
----------------------------------------------
报告
信息:有一客户端链接到本服务端
IP:user-PC
Port:7397
报告完毕
user-PC/192.168.30.223:7397 channelActive
defa23d9Tue Dec 30 16:54:51 CST 2014 群号:5360692
defa23d9Tue Dec 30 16:54:51 CST 2014 群号:5360692
defa23d9Tue Dec 30 16:54:52 CST 2014 群号:5360692
defa23d9Tue Dec 30 16:54:53 CST 2014 群号:5360692
报告
信息:有一客户端链接到本服务端
IP:localhost.localdomain
Port:7397
报告完毕
localhost.localdomain/127.0.0.1:7397 channelActive
5f735249Tue Dec 30 16:55:02 CST 2014 1
5f735249Tue Dec 30 16:55:03 CST 2014 1
5f735249Tue Dec 30 16:55:04 CST 2014 1
5f735249Tue Dec 30 16:55:04 CST 2014 1
5f735249Tue Dec 30 16:55:04 CST 2014 1
5f735249Tue Dec 30 16:55:05 CST 2014 1
5f735249Tue Dec 30 16:55:05 CST 2014 1
5f735249Tue Dec 30 16:55:05 CST 2014 1
5f735249Tue Dec 30 16:55:05 CST 2014 1
defa23d9Tue Dec 30 16:55:10 CST 2014 群号:5360692
----------------------------------------------
其中一个客户端控制输出:
---------------------------------------------
- 您已经开启与服务端链接 defa23d9Tue Dec 30 16:54:49 CST 2014 user-PC/192.168.30.223:7397
- 服务端收到:defa23d9Tue Dec 30 16:54:51 CST 2014 群号:5360692
- 服务端收到:defa23d9Tue Dec 30 16:54:51 CST 2014 群号:5360692
- 服务端收到:defa23d9Tue Dec 30 16:54:52 CST 2014 群号:5360692
- 服务端收到:defa23d9Tue Dec 30 16:54:53 CST 2014 群号:5360692
- 服务端收到:5f735249Tue Dec 30 16:55:02 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:03 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:04 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:04 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:04 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:05 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:05 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:05 CST 2014 1
- 服务端收到:5f735249Tue Dec 30 16:55:05 CST 2014 1
- 服务端收到:defa23d9Tue Dec 30 16:55:10 CST 2014 群号:5360692
- ---------------------------------------------
郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。