Linux客户端epoll,Linux非阻塞IO(七)使用epoll重新实现客户端

#define _GNU_SOURCE#include"sysutil.h"#include"buffer.h"#include

int main(int argc, char const *argv[])

{//创建client套接字

int sockfd = tcp_client(0);//调用非阻塞connect函数

int ret = nonblocking_connect(sockfd, "localhost", 9981, 5000);if(ret == -1)

{

perror("Connect Timeout .");

exit(EXIT_FAILURE);

}//将三个fd设置为Non-Blocking

activate_nonblock(sockfd);

activate_nonblock(STDIN_FILENO);

activate_nonblock(STDOUT_FILENO);

buffer_t recvbuf;//sockfd -> Buffer -> stdout

buffer_t sendbuf; //stdin -> Buffer -> sockfd//初始化缓冲区

buffer_init(&recvbuf);

buffer_init(&sendbuf);//创建epoll

int epollfd = epoll_create1(0);if(epollfd == -1)

ERR_EXIT("create epoll");struct epoll_event events[1024];

uint32_t sockfd_event= 0;

uint32_t stdin_event= 0;

uint32_t stdout_event= 0;

epoll_add_fd(epollfd, sockfd, sockfd_event);

epoll_add_fd(epollfd, STDIN_FILENO, stdin_event);

epoll_add_fd(epollfd, STDOUT_FILENO, stdout_event);while(1)

{//重新装填epoll事件

sockfd_event = 0;

stdin_event= 0;

stdout_event= 0;//epoll无法每次都重新装填,所以给每个fd添加一个空事件

if(buffer_is_readable(&sendbuf))

{

sockfd_event|=kWriteEvent;

}if(buffer_is_writeable(&sendbuf))

{

stdin_event|=kReadEvent;

}if(buffer_is_readable(&recvbuf))

{

stdout_event|=kWriteEvent;

}if(buffer_is_writeable(&recvbuf))

{

sockfd_event|=kReadEvent;

}

epoll_mod_fd(epollfd, sockfd, sockfd_event);

epoll_mod_fd(epollfd, STDIN_FILENO, stdin_event);

epoll_mod_fd(epollfd, STDOUT_FILENO, stdout_event);//监听fd数组

int nready = epoll_wait(epollfd, events, 1024, 5000);if(nready == -1)

ERR_EXIT("epoll wait");else if(nready == 0)

{

printf("epoll timeout.\n");continue;

}else{inti;for(i = 0; i < nready; ++i)

{int peerfd =events[i].data.fd;int revents =events[i].events;if(peerfd == sockfd && revents &kReadREvent)

{//从sockfd接收数据到recvbuf

if(buffer_read(&recvbuf, peerfd) == 0)

{

fprintf(stderr,"server close.\n");

exit(EXIT_SUCCESS);

}

}if(peerfd == sockfd && revents &kWriteREvent)

{

buffer_write(&sendbuf, peerfd); //将sendbuf中的数据写入sockfd

}if(peerfd == STDIN_FILENO && revents &kReadREvent)

{//从stdin接收数据写入sendbuf

if(buffer_read(&sendbuf, peerfd) == 0)

{

fprintf(stderr,"exit.\n");

exit(EXIT_SUCCESS);

}

}if(peerfd == STDOUT_FILENO && revents &kWriteREvent)

{

buffer_write(&recvbuf, peerfd); //将recvbuf中的数据输出至stdout

}

}

}

}

}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值