基于poll使用建立IO复用客户端

代码:

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <poll.h>

#define ERR_MSG(msg) do{\
    fprintf(stderr, "line:__%d__ ", __LINE__);\
    perror(msg);\
}while(0)

#define SER_PORT 8089          //服务器的端口
#define SER_IP  "192.168.8.168"  //服务器绑定的IP


int main(int argc, const char *argv[])
{
    //创建流式套接字
    int cfd = socket(AF_INET, SOCK_STREAM, 0);
    if(cfd < 0)
    {
        ERR_MSG("socket");
        return -1;
    }

   

    //填充要连接的服务器的IP和端口 
    struct sockaddr_in sin;
    sin.sin_family      = AF_INET;              //必须填AF_INET;
    sin.sin_port        = htons(SER_PORT);      //服务器绑定的端口号
    sin.sin_addr.s_addr = inet_addr(SER_IP);    //服务器绑定的IP地址

    //连接服务器
    if(connect(cfd, (struct sockaddr*)&sin, sizeof(sin)) < 0)
    {
        ERR_MSG("connect");
        return -1;
    }
    printf("connect server success\n");

    //定义要监测的文件描述符集合
    struct pollfd fds[2] = {0};

    fds[0].fd = 0;              //制定监测0号文件描述符  用于终端输入 并将消息发送给服务器
    fds[0].events = POLLIN;     //监测0号的读事件

    fds[1].fd = cfd;            //制定监测cfd     用于接收服务器消息
    fds[1].events = POLLIN;     //监测文件描述符的读事件


    char buf[128];
    ssize_t res = 0;
    int p_res = 0;                                                                                                        

    while(1)
    {
        p_res = poll(fds, 2, -1);
        if(p_res < 0)
        {
            ERR_MSG("poll");
            return -1;
        }
        else if(0 == p_res)
        {
            printf("time out ,,,,\n");
            break;
        }

        //能运行到当前位置,则代表集合中有文件描述符触发事件
        //需要判断集合中的文件描述符的revents中有没有POLLIN事件

        //实际产生的revents中可能有多个事件,POLLIN只是其中一个事件
        //所以需要通过按位与的方式,提取出POLLIN事件,判断有无POLLIN事件
        if(fds[0].revents & POLLIN)
        {
            //触发键盘输入事件
            bzero(buf, sizeof(buf)); //绑定客户端自身的地址信息结构体--->非必须绑定,可以选择不绑
    //如果选择不绑定,则操作系统会默认绑定一个端口号给这个客户端

            fgets(buf, sizeof(buf), stdin);
            buf[strlen(buf)-1] = 0;

            if(send(cfd, buf, sizeof(buf), 0) < 0)
            {
                ERR_MSG("send");
                return -1;
            }
            printf("send success\n");
        }






        if(fds[1].revents & POLLIN)
        {
            bzero(buf, sizeof(buf));
            res = recv(cfd, buf, sizeof(buf), 0);
            if(res < 0)
            {
                ERR_MSG("recv");
                return -1;
            }
            else if(0 == res)
            {
                printf("服务器端关闭\n");
                break;
            }

            printf(":%s\n", buf);
        }

    }


    //关闭文件描述符
    close(cfd);
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值