netty系列之:让TCP连接快一点,再快一点

简介: netty系列之:让TCP连接快一点,再快一点

简介


经典的TCP三次握手大家应该很熟悉了,三次握手按道理说应该是最优的方案了,当然这是对于通用的情况来说的。那么在某些特殊的情况下是不是可以提升TCP建立连接的速度呢?


答案是肯定的,这就是今天我们要讲的TCP fast open和netty。


TCP fast open


什么是TCP fast open呢?


TCP fast open也可以简写为TFO,它是TCP协议的一种扩展。为什么是fast open呢?这是因为TFO可以在初始化建立连接的时候就带上部分数据,这样在TCP连接建立之后,可以减少和服务器交互的次数,从而在特定的情况下减少响应的时间。


既然TFO这么好,为什么我们很少见到使用TFO协议的呢?


这是因为TFO是有缺陷的,因为TFO会在sync包中带上一些数据信息,那么当sync包重发的时候,就会造成接收方接受到重复的数据。


所以,如果是用TFO,那么接收方则需要具有能够处理重复数据的能力。


在程序界,防止数据重复提交有一个好听的名字叫做幂等性,只有具有幂等性的服务器才能够使用TFO。


开启TFO


既然TFO这么优秀,怎么才能开启TFO呢?


TFO的开启首先需要操作系统的支持,如果你是mac系统,恭喜你,mac默认情况下已经支持TFO了,你不需要进行任何操作。


如果你是Linux系统,那么需要查看/proc/sys/net/ipv4/tcp_fastopen这个文件。


tcp_fastopen可以有四种值,如下所示:


0 -- 表示TFO未开启

1 -- 表示TFO开启了,但是只对客户端有效

2 -- 表示TFO开启了,但是只对服务器端有效

3 -- 表示TFO开启了,同时对客户端和服务器端有效


通过上面的设置,我们就在操作系统层开启了TFO的支持。


接下来,我们看一下如何在netty中使用TFO。


netty对TFO的支持


首先我们看下如何在netty的服务器端开启TFO支持。


在这之前,我们先回顾一下如何建议一个普通的netty服务器:


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 TFOServerHandler());
                        }
                    })
                    .option(ChannelOption.SO_BACKLOG, 128)
                    .childOption(ChannelOption.SO_KEEPALIVE, true);
            // 绑定端口并开始接收连接
            ChannelFuture f = b.bind(port).sync();


上面的代码中,我们看到ServerBootstrap可以设置option参数,ChannelOption中包含了所有可以设置的channel的参数,对应的TFO的参数是ChannelOption.TCP_FASTOPEN, 所以我们只需要添加到ServerBootstrap中即可:


sb.option(ChannelOption.TCP_FASTOPEN, 50)


ChannelOption.TCP_FASTOPEN的值表示的是socket连接中可以处于等待状态的fast-open请求的个数。


对于客户端来说,同样需要进行一些改动,先来看看传统的client端是怎么工作的:


EventLoopGroup group = new NioEventLoopGroup();
        try {
            Bootstrap b = new Bootstrap();
            b.group(group)
             .channel(NioSocketChannel.class)
             .handler(new ChannelInitializer<SocketChannel>() {
                 @Override
                 protected void initChannel(SocketChannel ch) throws Exception {
                     ChannelPipeline p = ch.pipeline();
                     p.addLast(new TFOClientHandler());
                 }
             });
            // 连接服务器
            ChannelFuture f = b.connect(HOST, PORT).sync();


client要支持TFO,需要添加这样的操作:


b.option(ChannelOption.TCP_FASTOPEN_CONNECT, true)


还记得TFO是做什么的吗?TFO就是在sync包中发送了一些数据。所以我们需要在client端对发送的数据进行处理,也就是说在client和server端建立连接之前就需要向channel中发送消息。


要获得非建立连接的channel,则可以调用Bootstrap的register方法来获取channel:


Channel channel = b.register().sync().channel();


然后向该channel中写入byteBuf:


ByteBuf fastOpenData = directBuffer();
            fastOpenData.writeBytes("TFO message".getBytes(StandardCharsets.UTF_8));
            channel.write(fastOpenData);


最后再和服务器建立连接:


// 连接服务器
            SocketAddress serverAddress =  SocketUtils.socketAddress("127.0.0.1", 8000);
            ChannelFuture f = channel.connect(serverAddress).sync();


总结



这样一个一个支持TFO的客户端和服务器就完成了。尽情使用吧。

相关文章
|
8月前
|
网络协议 Java Maven
基于Netty实现TCP通信
基于Netty实现TCP通信
117 0
|
8月前
|
编解码 缓存 移动开发
TCP粘包/拆包与Netty解决方案
TCP粘包/拆包与Netty解决方案
113 0
|
3月前
|
网络协议 前端开发
netty的TCP服务端和客户端实现
本文介绍了使用Netty框架实现TCP服务端和客户端的步骤,包括添加Netty依赖、编写服务端和客户端的代码,涉及NioEventLoopGroup、ServerBootstrap、Bootstrap、ChannelInitializer等核心组件,以及如何启动服务端监听和客户端连接。
251 4
|
8月前
|
网络协议
Netty实现TCP通信
Netty实现TCP通信
113 0
|
8月前
|
网络协议 Java 物联网
Spring Boot与Netty打造TCP服务端(解决粘包问题)
Spring Boot与Netty打造TCP服务端(解决粘包问题)
1113 2
|
网络协议
netty编程实战02-创建一个带有连接重试的tcp客户端程序
netty编程实战02-创建一个带有连接重试的tcp客户端程序
221 0
|
8月前
|
前端开发 Java Maven
【Netty 网络通信】启动客户端连接服务端实现通信
【1月更文挑战第9天】【Netty 网络通信】启动客户端连接服务端实现通信
|
网络协议 算法
Netty入门到超神系列-TCP粘包拆包处理
TCP是面向连接的,服务端和客户端通过socket进行数据传输,发送端为了更有效的发送数据,通常会使用Nagle算法把多个数据块合并成一个大的数据块,这样做虽然提高了效率,但是接收端就很难识别完整的数据包了(TCP无消息保护边界),可能会出现粘包拆包的问题。
107 0
|
存储 缓存 编解码
一文搞定Netty,打造单机百万连接测试!3
一文搞定Netty,打造单机百万连接测试!
|
消息中间件 缓存 网络协议
使用 Netty+SpringBoot 打造的 TCP 长连接通讯方案 上
使用 Netty+SpringBoot 打造的 TCP 长连接通讯方案 上
使用 Netty+SpringBoot 打造的 TCP 长连接通讯方案 上