先打个广告,本人的微信公众号正式上线了,搜索:张笑生的地盘,主要关注嵌入式软件开发,股票基金定投,足球等等,希望大家多多关注,有问题可以直接留言给我,一定尽心尽力回答大家的问题
一 why
在前面的博客《linux进程间通信—本地socket套接字(四)—多路IO转接服务器实现一个server对应多个client》中,我们使用了多路IO转接实现了一个server对应多个client,我们使用的是select函数监听各个文件描述符的状态。
但是使用select函数有如下几个缺点:
1、select函数中文件描述符的个数是有限制的,最大是1024(当然我们可以修改这个限制,不过修改它需要我们重新编译内核,显然不是一个比较合适的方式)
2、在select函数中,待监听的文件描述符和监听到相关事件的文件描述符是同一个对象,而poll函数将两者做了区分,用两个变量来描述
3、select函数中需要搜索的范围更大,比如我们有文件描述符1,4,1000都检测到了读事件,select就需要遍历1000个(最大的,当然程序中可以使用一些处理技巧,只遍历这几个);而poll的范围就小了很多
二 what
poll函数原型:
int poll(struct pollfd *fds, nfds_t nfds, int timeout)
struct pollfd {
int fd; /* file descriptor */
short events; /* requested events */
short revents; /* returned events */
};
fds: 数组的首地址
nfds:数组的个数
timeout: 超时等待
————————————————
版权声明:本文为CSDN博主「拉玛西亚影视学院的奥斯卡影帝」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/myselfzhangji/article/details/104347032
我们可以设置poll函数中的文件描述符的最大值,设置方式如下
1. 用cat命令查看一个进程可以打开的socket描述符上限
cat /proc/sys/fs/file-max
2. 如果由需要,可以通过修改配置温恩建的方式修改该上限值
sudo vi /etc/security/limits.conf
在文件尾部写入以下配置,soft软限制,hard硬限制,如下图所示
soft nofile 65536
hard nofile 100000
————————————————
版权声明:本文为CSDN博主「拉玛西亚影视学院的奥斯卡影帝」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/myselfzhangji/article/details/104347032
fds的初始化如下:
fds[0].fd = listenfd
fds[0].events = POLLIN /POLLOUT/POLLERR,这个是用户设置的
fds[0].revents = POLLIN 这个是系统返回之后,帮我们填写上的,我们只需要判断这个值就知道对应的事件是否发生
三 how
接下来,我们尝试使用poll函数实现一个server对接多个client
先server.c代码
#include
再上client.c端代码
#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/shm.h>
#define PORT 8890
#define BUFFER_SIZE 1024
#define LOCAL_LOOP_BACK_IP "127.0.0.1"
int main(int argc, char **argv)
{
struct sockaddr_in servaddr;
char sendbuf[BUFFER_SIZE] = {0};
char recvbuf[BUFFER_SIZE] = {0};
int client_fd;
//定义IPV4的TCP连接的套接字描述符
client_fd = socket(AF_INET,SOCK_STREAM, 0);
//set sockaddr_in
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = inet_addr(LOCAL_LOOP_BACK_IP);
servaddr.sin_port = htons(PORT); //服务器端口
//连接服务器,成功返回0,错误返回-1
if (connect(client_fd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)
{
perror("connect");
exit(1);
}
printf("connect server(IP:%s).n",LOCAL_LOOP_BACK_IP);
//客户端将控制台输入的信息发送给服务器端,服务器原样返回信息
while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)
{
send(client_fd, sendbuf, strlen(sendbuf),0); ///发送
if(strcmp(sendbuf,"exitn")==0)
{
printf("client exited.n");
break;
}
recv(client_fd, recvbuf, sizeof(recvbuf),0); ///接收
printf("client receive: %sn", recvbuf);
memset(sendbuf, 0, sizeof(sendbuf));
memset(recvbuf, 0, sizeof(recvbuf));
}
close(client_fd);
return 0;
}