contentprovider实现跨进程通信_linux进程通信--本地socket套接字(五)--poll实现

先打个广告,本人的微信公众号正式上线了,搜索:张笑生的地盘,主要关注嵌入式软件开发,股票基金定投,足球等等,希望大家多多关注,有问题可以直接留言给我,一定尽心尽力回答大家的问题

e1011468f22da391ba3d94fb8fc54652.png

一 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;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值