功能清单
- 编写一个 NIO 群聊系统,实现客户端与客户端的通信需求(非阻塞)
- 服务器端:可以监测用户上线,离线,并实现消息转发功能
- 客户端:通过 channel 可以无阻塞发送消息给其它所有客户端用户,同时可以接受其它客户端用户通过服务端转发来的消息
服务端
- 定义成员属性:选择器、服务端通道、端口
- 初始化代码逻辑
- 获取通道
- 切换为非阻塞模式
- 绑定连接的端口
- 获取选择器Selector
- 将通道都注册到选择器上去,并且开始指定监听接收事件
- 监听事件
- 获取选择器中的所有注册的通道中已经就绪好的事件,
while (selector.select() > 0)
- 开始遍历这些准备好的事件
- 提取当前这个事件,判断这个事件具体是什么
- 客户端接入请求,
if(sk.isAcceptable())
- 直接获取当前接入的客户端通道,
ssChannel.accept()
- 切换成非阻塞模式,
schannel.configureBlocking(false);
- 将本客户端通道注册到选择器,
schannel.register(selector , SelectionKey.OP_READ);
- 客户端读请求,
if(sk.isReadable())
,声明并实现新方法private void readData(SelectionKey key)
- 得到当前客户端通道,
(SocketChannel) key.channel()
- 创建缓冲区对象开始接受客户端通道数据
- 提取读取到的信息,把缓存区的数据转成字符串
- 向其它的客户端转发消息(去掉自己), 专门写一个方法
sendInfoToOtherClients(String msg, SocketChannel self )
来处理
- 遍历 所有注册到selector 上的 SocketChannel
- 去除掉自己
- 将msg存到buffer,将buffer写进通道
- 如果离线了就关闭通道
- 处理完毕之后需要移除当前事件
- main方法调用
public class Server {
private Selector selector;
private ServerSocketChannel ssChannel;
private static final int PORT = 9999;
public Server() {
try {
ssChannel = ServerSocketChannel.open();
ssChannel.configureBlocking(false);
ssChannel.bind(new InetSocketAddress(PORT));
selector = Selector.open();
ssChannel.register(selector , SelectionKey.OP_ACCEPT);
}catch (IOException e) {
e.printStackTrace();
}
}
public void listen() {
System.out.println("监听线程: " + Thread.currentThread().getName());
try {
while (selector.select() > 0){
System.out.println("开始一轮事件处理~~~");
Iterator<SelectionKey> it = selector.selectedKeys().iterator();
while (it.hasNext()){
SelectionKey sk = it.next();
if(sk