前面几节我们讨论了非阻塞IO的基本概念、Buffer的设计以及非阻塞connect的实现,现在我们使用它们来完成客户端的编写。

我们在http://www.cnblogs.com/inevermore/p/4049165.html中提出过,客户端需要监听stdin、stdout和sockfd。

这里需要注意的是

只有缓冲区可写的时候,才去监听sockfd和stdin的读事件。

过去在阻塞IO中,我们总是监听sockfd的读事件,因为每当sockfd可读,我们就去调用用户的回调函数处理read事件,在回调函数中需要用户手工read缓冲区的数据。 换句话说,接收数据是用户的责任,poll模型只需要提醒用户去接收即可。

而在非阻塞IO中,因为poll采用的是水平触发,如果缓冲区满了,每次read等于无效操作,那么数据始终堆积在内核中,poll会不停的被触发。这在某种程度上等于轮询。所以我们只在缓冲区可用的情况下监听sockfd的读事件。

只有缓冲区可读的时候,才去监听sockfd和stdout的写事件。因为没有数据可写,监听write事件除了不停的触发poll之外,没有实际意义。

所以每次执行poll之前,需要重新装填poll的events数组。

完整的代码如下:

#include "sysutil.h"
#include "buffer.h" int main(int argc, char const *argv[])
{
//创建client套接字
int sockfd = tcp_client(8934);
//调用非阻塞connect函数
int ret = nonblocking_connect(sockfd, "192.168.44.136", 9981, 5000);
if(ret == -1)
{
fprintf(stderr, "Timeout .\n");
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); struct pollfd pfd[10]; while(1)
{
//初始化
int ix;
for(ix = 0; ix != 3; ++ix)
{
pfd[ix].fd = -1;
pfd[ix].events = 0;
} //重新装填events数组
if(buffer_is_readable(&sendbuf))
{
pfd[0].fd = sockfd;
pfd[0].events |= kWriteEvent;
}
if(buffer_is_writeable(&sendbuf))
{
pfd[1].fd = STDIN_FILENO;
pfd[1].events |= kReadEvent;
}
if(buffer_is_readable(&recvbuf))
{
pfd[2].fd = STDOUT_FILENO;
pfd[2].events |= kWriteEvent;
}
if(buffer_is_writeable(&recvbuf))
{
pfd[0].fd = sockfd;
pfd[0].events |= kReadEvent;
} //监听fd数组
int nready = poll(pfd, 3, 5000);
if(nready == -1)
ERR_EXIT("poll");
else if(nready == 0)
{
printf("timeout\n");
continue;
}
else
{
int i;
for(i = 0; i < 3; ++i)
{
int fd = pfd[i].fd;
if(fd == sockfd && pfd[i].revents & kReadEvent)
{
//从sockfd接收数据到recvbuf
if(buffer_read(&recvbuf, fd) == 0)
{
fprintf(stderr, "server close.\n");
exit(EXIT_SUCCESS);
}
} if(fd == sockfd && pfd[i].revents & kWriteEvent)
buffer_write(&sendbuf, fd); //将sendbuf中的数据写入sockfd if(fd == STDIN_FILENO && pfd[i].revents & kReadEvent)
{
//从stdin接收数据写入sendbuf
if(buffer_read(&sendbuf, fd) == 0)
{
fprintf(stderr, "exit.\n");
exit(EXIT_SUCCESS);
}
} if(fd == STDOUT_FILENO && pfd[i].revents & kWriteEvent)
buffer_write(&recvbuf, fd); //将recvbuf中的数据输出至stdout
}
}
} }

从以上的代码可以看出,大部分操作被封装进了buffer的实现中。

 

测试服务器,我暂时使用muduo库编写一个,代码如下:

#include <muduo/net/TcpServer.h>
#include <muduo/net/InetAddress.h>
#include <muduo/net/TcpConnection.h>
#include <muduo/base/Timestamp.h>
#include <muduo/net/EventLoop.h>
#include <muduo/base/Logging.h>
using namespace muduo;
using namespace muduo::net; void onMessage(const TcpConnectionPtr &conn, Buffer *buf, Timestamp t)
{
string s(buf->retrieveAllAsString());
LOG_INFO << "recv msg : " << s.size() << " at: " << t.toFormattedString();
conn->send(s);
} int main(int argc, char const *argv[])
{
EventLoop loop;
InetAddress addr("192.168.44.136", 9981);
TcpServer server(&loop, addr, "EchoServer");
server.setMessageCallback(&onMessage);
server.start(); loop.loop(); return 0;
}

读者如果使用上述的代码需要安装muduo网络库。

采用以下命令编译:

g++ server.cpp  -lmuduo_net -lmuduo_base -lpthread -o server

 

下文用poll实现非阻塞的服务器端。

最新文章

  1. KALI Linux problems &amp; Study Red Hat | Ubuntu
  2. keepalived mariadb 主主
  3. SQL server 链接查询
  4. MAC下 JDK环境配置、版本切换以及ADB环境配置
  5. [Android Pro] android Flag介绍
  6. ACM题目————Robot Motion
  7. hadoop2.5.1搭建(二)
  8. [网页设计]Ajax、Comet与Websocket--转
  9. Android 系统内置App JNI
  10. [NYOJ 43] 24 Point game
  11. #include&lt;iostream&gt;与#include&lt;iostream.h&gt;的区别
  12. centos 安装mysql密码修改后还是不能连接的原因
  13. hibernate利用mysql的自增张id属性实现自增长id和手动赋值id并存
  14. ●HDU 3507 Print Article
  15. 引用provinces.js的三级联动
  16. GCC 警告
  17. [luogu3628][bzoj1911][APIO2010]特别行动队【动态规划+斜率优化DP】
  18. java字符串转义,把&amp;lt;&amp;gt;转换成&lt;&gt;等字符【原】
  19. python数据类型之字典(二)
  20. CONTINUOUS MIGRATION

热门文章

  1. codevs 1269 匈牙利游戏——次短路(spfa)
  2. 64位操作系统安装32位客户端和PL/SQL
  3. 培训补坑(day5:最小生成树+负环判断+差分约束)
  4. linux 内存查看方法:meminfo\maps\smaps\status 文件解析
  5. Linux 设备驱动--- 阻塞型字符设备驱动 --- O_NONBLOCK --- 非阻塞标志【转】
  6. functools模块方法学习(1):partial
  7. C#集合类:动态数组、队列、栈、哈希表、字典(转)
  8. 阿里云服务器,tomcat启动,一直卡在At least one JAR was scanned for TLDs yet contained no TLDs就不动了
  9. Java android DES+Base64加密解密
  10. 模板:统计1~n内x的个数