Netty 创建服务端代码记录

Netty 创建服务端代码记录

/**
 * netty启动器
 */
@Slf4j
@Component
public class NettyServerBootstrap {

    private Channel serverChannel;
    private static final int DEFAULT_PORT = 9999;
    //bossGroup只是处理连接请求
    private static EventLoopGroup bossGroup = null;
    //workGroup处理非连接请求,如果牵扯到数据量处理业务非常耗时的可以再单独新建一个eventLoopGroup,并在childHandler初始化的时候添加到pipeline绑定
    private static EventLoopGroup workGroup = null;

    public boolean start() {
        return start(DEFAULT_PORT);
    }

    /**
     * 启动Netty服务
     *
     * @param port 端口
     * @return 启动结果
     */
    public boolean start(int port) {
        bossGroup = new NioEventLoopGroup();
        workGroup = new NioEventLoopGroup();
        //创建服务端启动对象
        ServerBootstrap bootstrap = new ServerBootstrap();
        try {
            //使用链式编程来设置
            bootstrap.group(bossGroup, workGroup)//设置两个线程组
                    //使用NioSocketChannel作为服务器的通道实现
                    .channel(NioServerSocketChannel.class)
                    //设置线程队列得到的连接数
                    .option(ChannelOption.SO_BACKLOG, 1024)
                    //设置保持活动连接状态
                    .childOption(ChannelOption.SO_KEEPALIVE, true)
                    //设置处理器  WorkerGroup 的 EvenLoop 对应的管道设置处理器
                    .childHandler(new ChildChannelHandler()); //创建通道初始化对象
            //绑定端口, 同步等待成功;
            ChannelFuture future = bootstrap.bind(port).sync();
            log.info("netty服务启动成功,ip:{},端口:{}", InetAddress.getLocalHost().getHostAddress(), port);
            serverChannel = future.channel();
            ThreadUtil.execute(() -> {
                //等待服务端监听端口关闭
                try {
                    future.channel().closeFuture().sync();
                    log.info("netty服务正常关闭成功,ip:{},端口:{}", InetAddress.getLocalHost().getHostAddress(), port);
                } catch (InterruptedException | UnknownHostException e) {
                    e.printStackTrace();
                } finally {
                    shutdown();
                }
            });
        } catch (Exception e) {
            e.printStackTrace();
            log.error("netty服务异常,异常原因:{}", e.getMessage());
            return false;
        }
        return true;
    }

    /***
     * 客户端处理器,用于初始化客户端pipeline,每次新的连接进来都会初始化一次
     */
    private static class ChildChannelHandler extends ChannelInitializer<SocketChannel> {
        @Override
        protected void initChannel(SocketChannel ch) throws Exception {
            log.info("初始化客户端id:{},远端ip:{}", ch.id(), ch.remoteAddress());
            //解决TCP报文粘包的问题
            ch.pipeline().addLast(new LengthFieldBasedFrameDecoder(ByteOrder.LITTLE_ENDIAN, Integer.MAX_VALUE, 2, 2, -4, 0, false));
            //解码
            ch.pipeline().addLast("decoder", new ByteArrayDecoder());
            //编码
            ch.pipeline().addLast("encoder", new ByteArrayEncoder());
            //给pipeline设置处理器
            ch.pipeline().addLast("nettyServerHandler", new NettyServerHandler());
        }
    }

    /**
     * 关闭当前server
     */
    public boolean close() {
        if (serverChannel != null) {
            serverChannel.close();//关闭服务
            try {
                //保险起见
                serverChannel.closeFuture().sync();
            } catch (InterruptedException e) {
                e.printStackTrace();
                return false;
            } finally {
                shutdown();
                serverChannel = null;
            }
        }
        return true;
    }

    /**
     * 优雅关闭
     */
    private void shutdown() {
        workGroup.shutdownGracefully();
        bossGroup.shutdownGracefully();
    }
/**
 * server handler
 */
@Slf4j
public class NettyServerHandler extends SimpleChannelInboundHandler<byte[]> {

    /**
     * 处理读取到的msg
     *
     * @param ctx 上下文
     * @param msg 数据
     */
    @Override
    protected void channelRead0(ChannelHandlerContext ctx, byte[] msg) throws Exception {
    }

    /**
     * 断开连接
     *
     * @param ctx 傻瓜下文
     */
    @Override
    public void handlerRemoved(ChannelHandlerContext ctx) throws Exception {
        ChannelId channelId = ctx.channel().id();
        log.info("客户端id:{},断开连接,ip:{}", channelId, ctx.channel().remoteAddress());
        super.handlerRemoved(ctx);
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的Netty WebSocket服务端代码示例: ```java import io.netty.bootstrap.ServerBootstrap; import io.netty.channel.Channel; import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelOption; import io.netty.channel.EventLoopGroup; import io.netty.channel.nio.NioEventLoopGroup; import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.nio.NioServerSocketChannel; import io.netty.handler.codec.http.HttpObjectAggregator; import io.netty.handler.codec.http.HttpServerCodec; import io.netty.handler.codec.http.websocketx.WebSocketServerProtocolHandler; public class WebSocketServer { private final int port; public WebSocketServer(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 HttpServerCodec(), new HttpObjectAggregator(65536), new WebSocketServerProtocolHandler("/websocket"), new WebSocketServerHandler()); } }) .option(ChannelOption.SO_BACKLOG, 128) .childOption(ChannelOption.SO_KEEPALIVE, true); ChannelFuture f = b.bind(port).sync(); System.out.println("WebSocketServer started on port " + port); f.channel().closeFuture().sync(); } finally { workerGroup.shutdownGracefully(); bossGroup.shutdownGracefully(); } } public static void main(String[] args) throws Exception { int port = 8080; if (args.length > 0) { port = Integer.parseInt(args[0]); } new WebSocketServer(port).run(); } } ``` 在这个代码中,我们创建了一个`WebSocketServer`类,它接受一个端口号作为参数,并在`run()`方法中启动Netty服务端。 在`run()`方法中,我们首先创建了两个`EventLoopGroup`,一个用于接收客户端连接的`bossGroup`,另一个用于处理客户端连接的`workerGroup`。 然后,我们创建了一个`ServerBootstrap`对象,并将`bossGroup`和`workerGroup`设置为它的组。我们还设置了`NioServerSocketChannel`作为通道类型。`childHandler()`方法用于设置`ChannelInitializer`,它将在每个新连接到达时创建一个新的`WebSocketServerHandler`实例。`WebSocketServerHandler`是我们实际处理WebSocket消息的类。 在`ChannelInitializer`中,我们将`HttpServerCodec`、`HttpObjectAggregator`和`WebSocketServerProtocolHandler`添加到管道中。`HttpServerCodec`用于将HTTP请求和响应编码和解码。`HttpObjectAggregator`用于将HTTP消息的多个部分合并为一个完整的`FullHttpRequest`或`FullHttpResponse`。`WebSocketServerProtocolHandler`用于将HTTP握手升级到WebSocket协议。 最后,我们使用`bind()`方法绑定端口,并启动服务端。我们还在`main()`方法中创建了一个`WebSocketServer`实例,并将其运行在指定的端口上。 这是一个简单的Netty WebSocket服务端代码示例,用于接收和处理WebSocket消息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值