写点什么

Netty 案例介绍 - 群聊案例实现,java 架构师教程百度云

作者:Java高工P7
  • 2021 年 11 月 10 日
  • 本文字数:2300 字

    阅读完需:约 8 分钟

  1. 编写一个 Netty 群聊系统,实现服务器端和客户端之间的数据简单通讯(非阻塞)

  2. 实现多人群聊

  3. 服务器端:可以监测用户上线,离线,并实现消息转发功能

  4. 客户端:通过 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

  • @de


《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》
浏览器打开:qq.cn.hn/FTe 免费领取
复制代码


scription: 群聊服务处理器


  • @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);


}


});


}


}

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();


}


}


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());


}


}

用户头像

Java高工P7

关注

还未添加个人签名 2021.11.08 加入

还未添加个人简介

评论

发布
暂无评论
Netty案例介绍-群聊案例实现,java架构师教程百度云