Linux网络编程之socket:由一个进程发起多个连接
来源:互联网 发布:雷欧团灭mac的怪兽 编辑:程序博客网 时间:2024/05/17 01:13
一、一个客户端即一个进程,只会发起一个连接,只要稍微修改一下就可以让一个客户端发起多个连
接,然后只利用其中一个连接发送数据。
首先了解一个函数getsocketname
#include <sys/socket.h> int getsockname(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
利用此函数可以得到某连接sockfd的地址信息,如ip地址和端口,这可以帮助我们判断发起了多少个连接。
我们假设一个客户端发起了5个连接,如下图:
此时根据以前说过的fork程序,服务器端会产生5个子进程对其进行服务。
客户端程序如下所示:
#include<stdio.h>#include<sys/types.h>#include<sys/socket.h>#include<unistd.h>#include<stdlib.h>#include<errno.h>#include<arpa/inet.h>#include<netinet/in.h>#include<string.h>#include "read_write.h"#define ERR_EXIT(m) \ do { \ perror(m); \ exit(EXIT_FAILURE); \ } while (0)void do_echocli(int sock){ char sendbuf[1024] = {0}; char recvbuf[1024] = {0}; while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL) { writen(sock, sendbuf, strlen(sendbuf)); int ret = readline(sock, recvbuf, sizeof(recvbuf)); //按行读取 if (ret == -1) ERR_EXIT("read error"); else if (ret == 0) //服务器关闭 { printf("server close\n"); break; } fputs(recvbuf, stdout); memset(sendbuf, 0, sizeof(sendbuf)); memset(recvbuf, 0, sizeof(recvbuf)); } close(sock);}int main(void){ int sock[5]; int i; for (i = 0; i < 5; i++) { if ((sock[i] = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0) // listenfd = socket(AF_INET, SOCK_STREAM, 0) ERR_EXIT("socket error"); struct sockaddr_in servaddr; memset(&servaddr, 0, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_port = htons(5188); servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); /* inet_aton("127.0.0.1", &servaddr.sin_addr); */ if (connect(sock[i], (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) ERR_EXIT("connect error"); struct sockaddr_in localaddr; socklen_t addrlen = sizeof(localaddr); if (getsockname(sock[i], (struct sockaddr *)&localaddr, &addrlen) < 0) ERR_EXIT("getsockname error"); /* getpeername()获取对等方的地址 */ printf("local ip=%s port=%d\n", inet_ntoa(localaddr.sin_addr), ntohs(localaddr.sin_port)); } /* 一个进程也可以发起多个socket连接,因为每次的端口号都不同 */ do_echocli(sock[0]); //发起5个套接字连接,但只借助第一个套接口通信 return 0;}在上述程序中,我们发起5个sock连接,但只是使用sock0通信,且利用getsockname 打印5个连接的信息。
先运行服务端程序
运行客户端程序
由于是多个连接,当客户端关闭而导致服务器子进程read 返回0退出进程时,很可能会产生僵尸进程,如下图:
最简单的办法就是父进程直接忽略SIGCHLD信号,即signal(SIGCHLD, SIG_IGN);
如果我们想要捕获SIGCHLD信号的话,在信号处理函数中不能只调用一次wait/waitpid 函数,因为客户端退出发出FIN段的时机是不一定的,如果都能按一定时间顺序发送给5个服务器子进程,即子进程发生SIGCHLD信号给父进程的时间有前后之分,那handler函数会被调用多次,则是允许的,也不会产生僵尸进程;但当多个SIGCHLD信号同时到达,因为不可靠信号不能排队导致信号只保存一个,即其余信号会丢失,则产生的僵尸进程个数是不确定的,因为按前面所说取决于5个SIGCHLD信号到达的次序。解决的办法很简单,只要在handler函数中while 循环一下就ok 了,即使5个信号同时到达,只要接收到一个SIGCHLD信号,则5个子进程都会被清理掉,如下所示:
signal(SIGCHLD, handler);.....................void handler(int sig){ /* wait(NULL); //只能等待第一个退出的子进程 */ while (waitpid(-1, NULL, WNOHANG) > 0) ;}
0 0
- Linux网络编程之socket:由一个进程发起多个连接
- linux网络编程之socket(七):一个进程发起多个连接和gethostbyname等函数
- linux网络编程之socket(七):一个进程发起多个连接和gethostbyname等函数
- linux网络编程之socket(七):一个进程发起多个连接和gethostbyname等函数
- linux网络编程之socket(七):一个进程发起多个连接和gethostbyname等函数
- linux网络编程之socket(七):一个进程发起多个连接和gethostbyname等函数
- Linux网络协议栈 -- socket connect 发起连接请求
- linux socket网络编程:fcntl select(多个客户端连接服务器端情形)
- linux socket网络编程:fcntl select(多个客户端连接服务器端情形)
- linux socket网络编程:fcntl select(多个客户端连接服务器端情形)
- linux socket网络编程:fcntl select(多个客户端连接服务器端情形)
- linux下socket编程实现一个服务器连接多个客户端
- Linux单机连接发起28232个连接
- linux网络编程之Socket
- linux socket网络编程之socket属性
- linux C 编程 之 socket 网络编程
- Linux网络编程之Socket编程
- linux网络编程之Socket编程原理
- Linux常用命令大全
- Go语言的10个实用技术
- Android中使用Handler造成内存泄露的分析和解决
- JasperReport + Servlet Web应用之:HTML显示报表内容
- java的MD5加密、解密
- Linux网络编程之socket:由一个进程发起多个连接
- Leetcode LRU Cache
- poj1094注意事项
- Oracle -- plsql异常处理
- Xcode6中自动布局autolayout和sizeclass的使用
- poj 2976 drop texts(二分搜索)
- fatal error LNK1123: 转换到 COFF 期间失败: 文件无效或损坏=》 解决办法
- 用calloc分配10块大小为4字节的内存空间,打印指针地址,并且打印出内存中的内容
- Linux权限管理-chown