java - Netty 服务器如何向不同的客户端发送数据?

标签 java netty

我想做一些简单的聊天。 服务器必须在客户端列表中添加新客户端,并且当一个客户端向服务器发送消息时,服务器必须将此消息重新发送给其他客户端。 我知道如何从客户端读取消息,但我不知道如何从服务器向客户端发送消息。而且我不确定客户列表应该放在哪里,但我猜是在处理程序类中。 这是我的主类,它初始化服务器类

package firstPackage;

public class main {

    public static void main(String[] args) throws Exception
    {
    Server server = new Server(9050);
    server.run();
    }
}

这是服务器类

package firstPackage;

import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioServerSocketChannel;

import io.netty.channel.socket.SocketChannel;


public class Server {

private int port;

public Server(int port)
{
    this.port=port;
}

public void run() throws Exception
{
    EventLoopGroup bossGroup = new NioEventLoopGroup();
    EventLoopGroup workerGroup = new NioEventLoopGroup();

    try{
        ServerBootstrap b = new ServerBootstrap();
        b.group(bossGroup,workerGroup)
         .channel(NioServerSocketChannel.class)
         .childHandler(new ChannelInitializer<SocketChannel>() {
             @Override
             public void initChannel(SocketChannel ch) throws Exception{
                 ch.pipeline().addLast(new DiscardServerHandler());
             }
         })
         .option(ChannelOption.SO_BACKLOG,128)
         .childOption(ChannelOption.SO_KEEPALIVE, true);

        ChannelFuture f = b.bind(port).sync();
        f.channel().closeFuture().sync();
    }
    finally {
        workerGroup.shutdownGracefully();
        bossGroup.shutdownGracefully();
    }
}

}

这里是 Handler 类

package firstPackage;

import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.util.ReferenceCountUtil;

public class DiscardServerHandler extends ChannelInboundHandlerAdapter {

@Override
public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
    ByteBuf in = (ByteBuf) msg;
    try {
        while (in.isReadable()) {
            System.out.print((char) in.readByte());
            System.out.flush();
        }
        System.out.println();

        ctx.writeAndFlush("hey"); // вот здесь я думал, что сообщение будет отправлятся клиенту, от которого я получил сообщение, но не отправляется

    } finally {
        ReferenceCountUtil.release(msg);
    }
}

@Override
public void channelActive(ChannelHandlerContext ctx) throws Exception {
    System.out.println("channel is active");
}

@Override
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
    System.out.println("channel is invactive");
}

@Override
public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
    System.out.println("handler added");
}

@Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
    cause.printStackTrace();
    ctx.close();
}
}

实际上我现在还没有客户端列表,因为我什至不知道这个列表必须包含什么类型的对象,在 C# 中是 Socket 对象,那么在 Netty 中呢?

最佳答案

我打赌this example应该可以帮到你(这是一个 YouTube 视频,所以请不要评论有关损坏的链接)。特别是,它使用 ChannelGroup解决你问的问题。而且,是的,它进入服务器端的处理程序。

编辑:

另请注意,示例中的 ChannelGroup 是静态的。我可能会争辩说,最好不要使用静态成员并将 ChannelGroup 从服务器类注入(inject)到处理程序中,但如果您只是想让某些东西工作,静态成员的简单性可能更可取很快。

关于java - Netty 服务器如何向不同的客户端发送数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26674866/

相关文章:

java - Android - 将数据添加到 SQL 中没有任何作用

java - GenericFilterBean vs OncePerRequestFilter 何时分别使用?

http - javax.net.ssl.SSLException : Inbound closed before receiving peer's close_notify: possible truncation attack?

java - 网络性能缓慢

java - 引用计数对象

java - 如何在同一集群上的 HBase 中创建表的副本?或者,如何在工作状态下操作时使用原始状态服务请求

Java - 单击按钮后绘制圆圈

java - mysqlexecuteBatch() 错误..?

Netty 4 (A1) - 具有多个处理程序的 AttributeMap

ssl - Play 框架看不到 SSL 中间/链证书