Linux网络编程中tcp_server和tcp_client函数的封装
本文的主要目的是将server套接字和client套接字的获取,做一个简易的封装,使用C语言完成。
tcp_server
服务器端fd的获取主要分为以下几步:
1.创建socket,这一步仅仅创建一个socket,没有任何特性的属性。
2.绑定网卡和port,一块主机可能有多块网卡,如果我们使用INADDR_ANY,意味着后面接受的TCP连接可以绑定在任意一块网卡上。
例如某台主机的ip地址有两个:192.168.44.136、10.1.1.4,假设绑定的ip采用INADDR_ANY,端口采用9981,那么当接收一个TCP连接时,可能存在192.168.44.136:9981/10.1.1.4:9981/127.0.0.1:9981三种可能性。
如果我们绑定192.168.44.136,那么这个fd接收的连接都是在这个ip上。
有一处特殊的地方,如果主机绑定的是127.0.0.1,那么客户端也必须是本机上的进程。(为什么?)
3.监听端口listen,实际上从这一步开始接收TCP连接。listen函数的第二个参数表示TCP待接受队列的大小,这个队列存储了已经完成三次握手,但还未被accept的连接。如果你设置为3,然后开启4个client,但是server端不进行accept,通过netstat –an | grep 9981,可以查看到只有三条server端的连接处于ESTABLISHED状态,其余的一个处于SYN_RCVD状态。
所以,很多人误以为accept完成三次握手,这是一种错误的想法,三次握手是由底层的协议栈完成的。
然后我们需要使用几个套接字选项,后面会专门写一篇文章总结套接字选项:
1.地址复用SO_REUSEADDR,代码如下:
//地址复用
void set_reuseaddr(int sockfd, int optval)
{
int on = (optval != 0) ? 1 : 0;
if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)
ERR_EXIT("setsockopt SO_REUSEADDR");
}
2.端口复用SO_REUSEPORT,有的机器不支持,所以需要判断
void set_reuseport(int sockfd, int optval)
{
#ifdef SO_REUSEPORT
int on = (optval != 0) ? 1 : 0;
if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEPORT, &on, sizeof(on)) < 0)
ERR_EXIT("setsockopt SO_REUSEPORT");
#else
fprintf(stderr, "SO_REUSEPORT is not supported.\n");
#endif //SO_REUSEPORT
}
3.禁用Nagle算法,降低局域网的延迟
//设置nagle算法是否可用
void set_tcpnodelay(int sockfd, int optval)
{
int on = (optval != 0) ? 1 : 0;
if(setsockopt(sockfd, IPPROTO_TCP, TCP_NODELAY, &on, sizeof(on)) == -1)
ERR_EXIT("setsockopt TCP_NODELAY");
}
4.禁用keepAlive机制
void set_keepalive(int sockfd, int optval)
{
int on = (optval != 0) ? 1 : 0;
if(setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE, &on, sizeof(on)) == -1)
ERR_EXIT("setsockopt SO_KEEPALIVE");
}
我们还需要为我们的tcp_server增加域名解析功能,我们可以绑定”localhost”之类的主机名,而不仅仅是ip地址,所以我们可能使用gethostbyname。
所以tcp_server代码如下:
int tcp_server(const char *host, uint16_t port)
{ //处理SIGPIPE信号
handle_sigpipe(); int listenfd = socket(PF_INET, SOCK_STREAM, 0);
if(listenfd == -1)
ERR_EXIT("socket"); set_reuseaddr(listenfd, 1);
set_reuseport(listenfd, 1);
set_tcpnodelay(listenfd, 0);
set_keepalive(listenfd, 0); SAI addr;
memset(&addr, 0, sizeof addr);
addr.sin_family = AF_INET;
//addr.sin_addr.s_addr = inet_addr("127.0.0.1");
addr.sin_port = htons(port);
if(host == NULL)
{
addr.sin_addr.s_addr = INADDR_ANY;
}
else
{
//int inet_aton(const char *cp, struct in_addr *inp);
if(inet_aton(host, &addr.sin_addr) == 0)
{
//DNS
//struct hostent *gethostbyname(const char *name);
struct hostent *hp = gethostbyname(host);
if(hp == NULL)
ERR_EXIT("gethostbyname");
addr.sin_addr = *(struct in_addr*)hp->h_addr;
}
} if(bind(listenfd, (SA*)&addr, sizeof addr) == -1)
ERR_EXIT("bind"); if(listen(listenfd, SOMAXCONN) == -1)
ERR_EXIT("listen"); return listenfd;
}
tcp_client
客户端也可以绑定port,只不过它不是必须的。代码较简单,如下:
int tcp_client(uint16_t port)
{
int peerfd = socket(PF_INET, SOCK_STREAM, 0);
if(peerfd == -1)
ERR_EXIT("socket"); set_reuseaddr(peerfd, 1);
set_reuseport(peerfd, 1);
set_keepalive(peerfd, 0);
set_tcpnodelay(peerfd, 0); //如果port为0,则不去绑定
if(port == 0)
return peerfd; SAI addr;
memset(&addr, 0, sizeof addr);
addr.sin_family = AF_INET;
addr.sin_port = htons(port);
addr.sin_addr.s_addr = inet_addr(get_local_ip());
if(bind(peerfd, (SA*)&addr, sizeof(addr)) == -1)
ERR_EXIT("bind client"); return peerfd;
}
上面需要获取本机的ip地址,代码如下:
const char *get_local_ip()
{
static char ip[16]; int sockfd;
if((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1)
{
ERR_EXIT("socket");
} struct ifreq req;
bzero(&req, sizeof(struct ifreq));
strcpy(req.ifr_name, "eth0");
if(ioctl(sockfd, SIOCGIFADDR, &req) == -1)
ERR_EXIT("ioctl"); struct sockaddr_in *host = (struct sockaddr_in*)&req.ifr_addr;
strcpy(ip, inet_ntoa(host->sin_addr));
close(sockfd); return ip;
}
完毕。
Linux网络编程中tcp_server和tcp_client函数的封装的更多相关文章
- linux网络编程中INADDR_ANY的含义
INADDR_ANY选项 网络编程中常用到bind函数,需要绑定IP地址,这时可以设置INADDR_ANY INADDR_ANY就是指定地址为0.0.0.0的地址,这个地址事实上表示不确定地址,或&q ...
- linux网络编程中的shutdown()与close()函数
1.close()函数 int close(int sockfd); //返回成功为0,出错为-1 close 一个套接字的默认行为是把套接字标记为已关闭,然后立即返回到调用进程,该套接字不能再由cl ...
- Linux 网络编程中的read和write函数正确的使用方式
字节流套接字上的read和write函数所表现的行为不同于通常的文件IO,字节流套接字上调用read和write输入或输出的可能比请求的数量少,然而这不是出错的状态,例如某个中端使read和write ...
- linux网络编程中的超时设置
1 下面是在网上找到的资料,先非常的感谢. 用setsockopt()来控制recv()与send()的超时 在send(),recv()过程中有时由于网络状况等原因,收发不能预期进行,而设置收发超时 ...
- linux网络编程中的基本概念
int close(int fd)(假设是服务器端) close 关闭了自身数据传输的两个方向.close一个TCP套接字的默认行为是把该套接字标记成已关闭,然后立即返回到调用进程.该套接字描述符不能 ...
- linux网络编程中阻塞和非阻塞socket的区别
读操作 对于阻塞的socket,当socket的接收缓冲区中没有数据时,read调用会一直阻塞住,直到有数据到来才返 回.当socket缓冲区中的数据量小于期望读取的数据量时,返回实际读取的字节数.当 ...
- linux网络编程中需要注意的信号SIGPIPE
在调试cs时,s端循环收,c端循环发,s端意外崩溃后,c端自动退出,终端提示SIGPIPE导致c端退出.man 7 signal: SIGPIPE Term Broken pipe: write to ...
- Linux网络编程入门 (转载)
(一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端 网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户 ...
- [转] - Linux网络编程 -- 网络知识介绍
(一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端 网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户 ...
随机推荐
- Topcoder SRM 605 div1 题解
日常打卡- Easy(250pts): 题目大意:你有n种汉堡包(统统吃掉-),每一种汉堡包有一个type值和一个taste值,你现在要吃掉若干个汉堡包,使得它们taste的总和*(不同的type值的 ...
- 汕头市队赛SRM14 T3覆盖
我们可以考虑两种情况 区间之间不相重叠 和 重叠 f[i][j]表示以当前最后一个区间以 i 结尾 并且选了 j 个区间 不相重叠的话 只要选 1-i-w 的max再加上 包含i在内的前四个数的和 相 ...
- EF 创建数据库的策略 codefist加快效率!【not oringin!】
今天去搜寻,ef创建数据库的策略有四种,区分还是和数据库里sql的创建的语句这些英文差不多一致. 一:数据库不存在时重新创建数据库 Database.SetInitializer<testCon ...
- UVA 10594 Data Flow
无向图费用流 还有一段话摘自别人博客 这道题是无向图的最小费用最大流问题,看清楚是无向图的.这么说无向图和有向图的费用流问题有什么区别呢?主要是反向边的问题.首先我们说一下最大流问题中的反向边,我们需 ...
- (七)insmod/rmmod
insmod: insmod命令用于将给定的模块加载到内核中.Linux有许多功能是通过模块的方式,在需要时才载入kernel.如此可使kernel较为精简,进而提高效率,以及保有较大的弹性.这类可载 ...
- ubuntu16下安装telnet和opensshserver
安装了虚拟机,使用的是ubuntu 16,server版本. 启动后发现没有telnet和ssh,就安装了(netstat -a|grep telnet). apt-get install openb ...
- (1)C#工具箱-公共控件1
公共控件 InitializeComponent() 先说下InitializeComponent()这个方法,它在form1.cs里调用这个方法对控件进行初始化,控件的方法要在这个方法之后,否则会因 ...
- HDU 6395 2018 Multi-University Training Contest 7 (快速幂+分块)
原题地址 Sequence Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)T ...
- [HDU5528]Count a * b
题目大意: 定义函数$f(m)=\displaystyle\sum_{a=0}^{m-1}\sum_{b=0}^{m-1}[m\nmid ab]$,$g(n)=\displaystyle\sum_{m ...
- Mobius反演与积性函数前缀和演学习笔记 BZOJ 4176 Lucas的数论 SDOI 2015 约数个数和
下文中所有讨论都在数论函数范围内开展. 数论函数指的是定义域为正整数域, 且值域为复数域的函数. 数论意义下的和式处理技巧 因子 \[ \sum_{d | n} a_d = \sum_{d | n} ...