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. Mac brew命令的使用

    mac 终端程序管理工具 能让你更快速的安装你想要的工具.而不用考虑大量的依赖. 安装brew复制下面的命令,终端执行  官网Homebrew /usr/bin/ruby -e "$(cur ...

  2. C# 基础知识系列- 4 面向对象

    面向对象 面向对象是一个抽象的概念,其本质就是对事物以抽象的方式建立对应的模型. 简单来讲,比如我有一只钢笔,那么我就可以通过分析,可以得到 这只钢笔的材第是塑料,品牌是个杂牌 ,里面装的墨是黑色的, ...

  3. IdentityServer4源码解析_5_查询用户信息接口

    协议简析 UserInfo接口是OAuth2.0中规定的需要认证访问的接口,可以返回认证用户的声明信息.请求UserInfo接口需要使用通行令牌.响应报文通常是json数据格式,包含了一组claim键 ...

  4. 关于js中iframe 中 location.href的用法

    关于js中"window.location.href"."location.href"."parent.location.href".&qu ...

  5. 怎样设计最优的卷积神经网络架构?| NAS原理剖析

    虽然,深度学习在近几年发展迅速.但是,关于如何才能设计出最优的卷积神经网络架构这个问题仍在处于探索阶段. 其中一大部分原因是因为当前那些取得成功的神经网络的架构设计原理仍然是一个黑盒.虽然我们有着关于 ...

  6. 《闲扯Redis三》Redis五种数据类型之List型

    一.前言 Redis 提供了5种数据类型:String(字符串).Hash(哈希).List(列表).Set(集合).Zset(有序集合),理解每种数据类型的特点对于redis的开发和运维非常重要. ...

  7. 高并发解决方案限流技术-----使用RateLimiter实现令牌桶限流

    1,RateLimiter是guava提供的基于令牌桶算法的实现类,可以非常简单的完成限流特技,并且根据系统的实际情况来调整生成token的速率.通常可应用于抢购限流防止冲垮系统:限制某接口.服务单位 ...

  8. 关于visocode 自动保存时自动添加分号问题

    先下载插件:  Vueter 打开设置的配置文件,写入一下代码: // //是否需要保存时自动格式化 "editor.formatOnSave": true, //使js 文件保存 ...

  9. Activiti任务分配

    分配任务负责人 一.固定分配 在进行业务流程建模时指定固定的任务负责人 在properties 视图中,填写Assignee 项为任务负责人. 注意: 由于固定分配方式,任务只管一步一步执行任务,执行 ...

  10. Windows 7集成IE11(离线安装包、补丁)

    当Win7系统需要集成IE11时,我们需要提前打入6个补丁 KB2731771.KB2786081.KB2834140.KB2670838.KB2729094.KB2533623 32位 ★百度网盘 ...