Linux C++ 网络编程学习系列(1)——端口复用实现

  1. 源码地址:https://github.com/whuwzp/linuxc/tree/master/portreuse
  2. 源码说明:
    • server1.cpp: 监听127.1:6666,功能是将收到的小写转大写
    • server2.cpp: 监听192.132:6666, 功能是接收数据, 将自己的数据从大写转小写, 把不是自己的数据转发给server1处理(判断方法是头三个字符是不是123)
    • client.cpp: 客户端, 地址在: https://github.com/whuwzp/linuxc/tree/master/simple_cs/client/

1. 概要

起因: 由socket状态转换图可知, 主动关闭连接的一端都会有一个TIME_WAIT, 时间为2msl, 以确保对端收到最后一个ACK

影响: 如果是服务器端需要主动断开连接(例如网站更新等), 那么再次重启则需要浪费2msl时间, 为了减少这种时间开销, 因此有了端口复用

作用: 可以立马重启服务器, 而不必等待2msl, 且不会在bind的时候发生端口被占用的错误.

2. 核心代码

2.1 server1

//server1.cpp

#define LOCALIP "127.0.0.1"
#define PORT 6666 // 127.1:6666. startsock里面很简单,就是socket函数
startsock(fd_server, sock_server, LOCALIP, PORT);
opt = 1;
//设置为端口复用
Setsockopt(fd_server, SOL_SOCKET, SO_REUSEADDR, &opt, (socklen_t)sizeof(opt));

2.2 server2

//server2.cpp

#define LOCALIP "127.0.0.1"
#define PUBLICIP "192.168.153.132"
#define myflag "123"
#define PORT 6666 //这是连接server1, 也就是127.1:6666
startsock(fd_local, sock_local, LOCALIP, PORT);
opt = 1;
Setsockopt(fd_local, SOL_SOCKET, SO_REUSEADDR, &opt, (socklen_t)sizeof(opt)); //这是绑定192.132:6666的公网地址
startsock(fd_server, sock_server, PUBLICIP, PORT);
opt = 1;
Setsockopt(fd_server, SOL_SOCKET, SO_REUSEADDR, &opt, (socklen_t)sizeof(opt)); while (true) {
ret = (int)Read(fd_client, recvbuf, sizeof(recvbuf));
if (ret == 0) {
break;
}
strcpy(cmpstr, recvbuf);
cmpstr[strlen(myflag)] = '\0';
/*看看是不是我的数据包,如果不是,则交给127.1:6666端口处理*/
if (strcmp(cmpstr, myflag) != 0) {
printf("going to send to local\n");
Write(fd_local, recvbuf, strlen(recvbuf) + 1);//发给server1处理
Read(fd_local, sendbuf, sizeof(sendbuf)); //接收server1的返回结果
Write(fd_client, sendbuf, strlen(sendbuf) + 1);//把返回结果转给客户
continue;
} //是我的,我来处理
printf("recv from %s:%d : string: %s\n",
inet_ntoa(clientinfo->addr.sin_addr),
ntohs(clientinfo->addr.sin_port), recvbuf); handler(recvbuf + strlen(myflag), sendbuf);//函数功能: 将大写转为小写
ret = (int)Write(fd_client, sendbuf, strlen(sendbuf) + 1);
if (ret == 0) {
break;
}
}

2.3 运行结果

# client
input:hello # 输入hello
send : 6 nbytes
recv: 6 bytes, string: HELLO # 由server1处理了
input:123HEllo # 输入123HEllo
send : 9 nbytes
recv: 6 bytes, string: hello # 由server2处理了 # server1
accept: 127.0.0.1: 55526
recv from 127.0.0.1:55526 : string: hello # server2
accept: 192.168.153.132: 55558
going to send to local
recv from 192.168.153.132:55558 : string: 123HEllo

3. 参考网址

参考视频:https://www.bilibili.com/video/av53016117?p=48

参考网址:https://www.cnblogs.com/royi123/archive/2013/03/12/2956655.html

Linux C++ 网络编程学习系列(1)——端口复用实现的更多相关文章

  1. Linux C++ 网络编程学习系列(7)——mbedtls编译使用

    mbedtls编译使用 环境: Ubuntu18.04 编译器:gcc或clang 编译选项: 静态编译使用 1. mbedtls源码 下载地址: https://github.com/ARMmbed ...

  2. Linux C++ 网络编程学习系列(6)——多路IO之epoll高级用法

    poll实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/epoll_libevent 源码说明: server.cpp: 监听127. ...

  3. Linux C++ 网络编程学习系列(5)——多路IO之epoll边沿触发

    多路IO之epoll边沿触发+非阻塞 源码地址:https://github.com/whuwzp/linuxc/tree/master/epoll_ET_LT_NOBLOCK_example 源码说 ...

  4. Linux C++ 网络编程学习系列(4)——多路IO之epoll基础

    epoll实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/epoll 源码说明: server.cpp: 监听127.1:6666,功 ...

  5. Linux C++ 网络编程学习系列(3)——多路IO之poll实现

    poll实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/poll 源码说明: server.cpp: 监听127.1:6666,功能是 ...

  6. Linux C++ 网络编程学习系列(2)——多路IO之select实现

    select实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/select 源码说明: server.cpp: 监听127.1:6666 ...

  7. Linux C网络编程学习笔记

    Linux C网络编程总结报告 一.Linux C 网络编程知识介绍: 网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户端:(client) 在网络程序中, ...

  8. linux下网络编程学习——入门实例ZZ

    http://www.cppblog.com/cuijixin/archive/2008/03/14/44480.html 是不是还对用c怎么实现网络编程感到神秘莫测阿,我们这里就要撕开它神秘的面纱, ...

  9. Linux下网络编程学习杂记

    1.TCP/IP协议的体系结构包含四层:应用层(负责应用程序的网络服务,通过端口号识别各个不同的进程)->传输层(传输控制层协议TCP.用户数据报协议UDP.互联网控制消息协议ICMP)-> ...

随机推荐

  1. 开源项目OEIP 游戏引擎与音视频多媒体(UE4/Unity3D)

    现开源一个项目 OEIP 项目实现的功能Demo展示 这个项目演示了在UE4中,接入摄像机通过OEIP直接输出到UE4纹理上,并直接把UE4里的RenderTarget当做输入源通过OEIP里GPU管 ...

  2. 测试leader的职责

    为了项目过程管理更加紧凑,控制项目测试进度.测试质量, 需要指定一个测试leader,测试leader的主要职责是测试边界的划定,整体进度的把控,项目风险识别和应对.具体工作内容如下: 一.需求阶段: ...

  3. GBDT梯度提升树算法及官方案例

    梯度提升树是一种决策树的集成算法.它通过反复迭代训练决策树来最小化损失函数.决策树类似,梯度提升树具有可处理类别特征.易扩展到多分类问题.不需特征缩放等性质.Spark.ml通过使用现有decisio ...

  4. Redis 缓存更新一致性

    当执行写操作后,需要保证从缓存读取到的数据与数据库中持久化的数据是一致的,因此需要对缓存进行更新. 因为涉及到数据库和缓存两步操作,难以保证更新的原子性. 在设计更新策略时,我们需要考虑多个方面的问题 ...

  5. Tensorflow 介绍和安装

    欢迎大家关注我们的网站和系列教程:http://www.tensorflownews.com/,学习更多的机器学习.深度学习的知识! 作者:AI小昕 本系列教程将手把手带您从零开始学习Tensorfl ...

  6. PHP7内核(六):变量之zval

    记得网上流传甚广的段子"PHP是世界上最好的语言",暂且不去讨论是否言过其实,但至少PHP确实有独特优势的,比如它的弱类型,即只需要$符号即可声明变量,使得PHP入手门槛极低,成为 ...

  7. RabbitMQ 交换机类型

    1,扇形交换机 fanout 2, 直连交换机 direct 3, 通配符交换机 topic

  8. JQuery主要内容

    一.什么是JQuery jquery全称javaScript Query,是js的一个框架,本质上仍然是js 二.jQuery的特点 支持各种主流浏览器 使用特别简单 拥有丰富的插件和边界的插件扩展机 ...

  9. 【cs224w】Lecture 4 - 社区结构

    Community 转自本人:https://blog.csdn.net/New2World/article/details/105328390 之前讲到了网络中节点扮演不同角色,而角色这个概念和社区 ...

  10. C Cow XOR 奶牛异或

    时间限制 : 10000 MS   空间限制 : 65536 KB 问题描述 农民约翰在喂奶牛的时候被另一个问题卡住了.他的所有N(1 <= N <= 100,000)个奶牛在他面前排成一 ...