【linux高级程序设计】(第十四章)TCP高级应用 2
socket多路复用应用
int select (int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout)
功能:轮循等待的方式,从多个文件描述符中获取状态变化后的情况
readfds :包含所有可能因状态变成可读而触发select()函数返回的文件描述符
writefds :包含所有可能因状态变成可写而触发select()函数返回的文件描述符
exceptfds :包含所有可能因状态发生特殊异常(如带外数据到来)而触发select()函数返回的文件描述符
针对文件描述符集合的操作如下:
#define FD_SET(fd, fdsetp) //把fd添加到fdsetp中
#define FD_CLR(fd, fdsetp) //从fdsetp中删除fd
#define FD_ISSET(fd, fdsetp) //检测fdsetp中的fd是否出现异常
#define FD_ZERO(fdsetp) //初始化fdsetp为空
参数1:限制上面要检测的文件描述符的范围,范围在0到最大文件描述符值之间
最后一个参数:表示阻塞超时时限
struct timeval {
long tv_sec;
long tv_usec;
};
返回值:函数错误,返回-1; 超时返回0,将时间结构体清空为0;有文件需要处理,返回相应的文件描述符,在文件描述符集合中清除不需要处理的文件描述符
例子:
1.检测某个socket是否可读
fd_set rdfds; //声明一个fd_set集合来保存要检测的socket
struct timeval tv; //保存时间
int ret; //保存返回值
FD_ZERO(&rdfds); //集合清零
FD_SET(socket, &rdfds); //把要检测的文件描述符加入集合
tv.tv_sec = ;
tv.tv_usec = ; //设置select等待的最大时间为1s+500ms
ret = select(socket + , &rdfds, NULL, NULL, &tv); //检测集合中是否有可读信息
if(ret < ) //出错
perror("select");
else if(ret == ) //超时
printf("超时\n");
else //有状态变化
{
printf("ret = %d\n", ret);
//判断socket是否变成可读
if(FD_ISSET(socket, &rdfds))
{
recv(...); //读取
}
}
2.检测用户键盘输入。需要把标准输入文件描述符0放入select检测
FD_ZERO(&rdfds);
FD_SET(, &rdfds);
tv.tv_sec = ;
tv.tv_usec = ;
ret = select(, &rdfds, NULL, NULL, &tv);
if(ret < ) //出错
perror("select");
else if(ret == ) //超时
printf("超时\n");
else //有输入
scanf("%s", buf);
int pselect (int __nfds, fd_set *__restrict __readfds, fd_set *__restrict __writefds, fd_set *__restrict __exceptfds, const struct timespec *__restrict __timeout, const __sigset_t *__restrict __sigmask)
该函数与select()函数功能几乎相同,只是时间精度更高,同时设置了阻塞的信号集合。
时间的结构体声明如下:
struct timespec{
long ts_sec;
long ts_nsec; //ns
};
poll与ppoll函数
可以实现比select/pselect函数更强大的功能,更细粒的等待时间
int poll (struct pollfd *fds, nfds_t nfds, int timeout)
int ppoll (struct pollfd *fds, nfds_t nfds, const struct timespec *timeout_ts, const sigset_t *sigmask)
其中文件描述符的结构体定义为:
struct pollfd{
int fd; //文件描述符
short events; //请求事件
short revents; //返回的事件
};
请求或返回的事件类型如下:
ppoll()函数也可以在阻塞过程中屏蔽某些信号,而且timeout上ppoll的精度更高。
调用
ready = ppoll(&fds, nfds, timeout_ts, &sigmask);
相当于调用
sigset_t origmask;
int timeout;
timeout = (timeout_ts == NULL) ? - : (timeout_ts.tv_sec * + timeout_ts.tv_nesc / );
sigprocmask(SIG_SETMASK, &sigmask, &origmask);
ready = poll(&fds, nfds, timeout);
sigprocmask(SIG_SETMASK, &origmask, NULL);
示例
基于多路复用的服务器客户端聊天程序
这个的效果是目前为止最好的,可以实现一对多的通信。消息也比较清晰。
服务器端
#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#include<string.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<sys/socket.h>
#include<sys/wait.h>
#include<unistd.h>
#include<arpa/inet.h>
#include<sys/time.h>
#define MAXBUF 1024
int main(int argc, char * argv[])
{
int sockfd, new_fd;
socklen_t len;
struct sockaddr_in my_addr, their_addr;
unsigned int myport, lisnum;
char buf[MAXBUF + ];
fd_set rfds; //文件描述符集合
struct timeval tv;
int retval, maxfd = -;
if(argv[])
myport = atoi(argv[]); //参数2为端口号
else
myport = ; //默认端口号
if(argv[])
lisnum = atoi(argv[]); //命令行第3个参数为listen队列大小 即可以等待多少个客户端
else
lisnum = ;
//创建socket对象 ipv4 TCP 默认协议
if((sockfd = socket(PF_INET, SOCK_STREAM, )) == -)
{
perror("socket");
exit(EXIT_FAILURE);
}
bzero(&my_addr, sizeof(my_addr));
my_addr.sin_family = PF_INET;
my_addr.sin_port = htons(myport);
if(argv[])
my_addr.sin_addr.s_addr = inet_addr(argv[]); //参数1为IP地址
else
my_addr.sin_addr.s_addr = INADDR_ANY;
//绑定地址信息
if(bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) == -)
{
perror("bind");
exit(EXIT_FAILURE);
}
//服务器监听网络
if(listen(sockfd, lisnum) == -)
{
perror("listen");
exit(EXIT_FAILURE);
}
while()
{
printf("\n---------wait for new connect\n");
len = sizeof(struct sockaddr);
//接收客户端连接
if((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, &len)) == -)
{
perror("accept");
exit(EXIT_FAILURE);
}
else
{
//打印连接信息
printf("server: got connection from %s, port %d, socket %d\n", inet_ntoa(their_addr.sin_addr), ntohs(their_addr.sin_port), new_fd);
while()
{
FD_ZERO(&rfds);
FD_SET(, &rfds);
FD_SET(new_fd, &rfds);
maxfd = new_fd; //只有两个文件描述符0和new_fd,最大值为sockfd
tv.tv_sec = ;
tv.tv_usec = ;
//多路复用
retval = select(maxfd + , &rfds, NULL, NULL, &tv);
if(retval == -) //函数出错
{
perror("select");
exit(EXIT_FAILURE);
}
else if(retval == ) //超时
{
continue;
}
else
{
//检测是否为标准输入引起异常
if(FD_ISSET(, &rfds))
{
bzero(buf, MAXBUF + );
fgets(buf, MAXBUF, stdin); //从标准输入读数据
if(!strncasecmp(buf, "quit", )) //如果quit退出
{
printf("i will quit!\n");
break;
}
//将数据发送给客户端
len = send(new_fd, buf, strlen(buf) - , ); //为什么要-1 ??
if(len > )
printf("send successful, %d byte send!\n", len);
else
{
printf("send failure!");
break;
}
}
//如果是当前sockfd引起的异常
if(FD_ISSET(new_fd, &rfds))
{
bzero(buf, MAXBUF + );
//从中读取数据
len = recv(new_fd, buf, MAXBUF, );
if(len > )
printf("recv success :'%s', %dbyte recv\n", buf, len);
else if(len == )
{
printf("the other one end quit\n");
break;
}
}
}
}
}
close(new_fd);
printf("need other connect (no->quit)"); //是否需要等待其他客户端连接
fflush(stdout); //刷新标准输出
bzero(buf, MAXBUF + );
fgets(buf, MAXBUF, stdin);
if(!strncasecmp(buf, "no", ))
{
printf("quit!\n");
break;
}
}
close(sockfd);
return ;
}
客户端
#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#include<string.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<sys/socket.h>
#include<sys/wait.h>
#include<unistd.h>
#include<arpa/inet.h>
#include<sys/time.h>
#define MAXBUF 1024
int main(int argc, char **argv)
{
int sockfd, len;
struct sockaddr_in dest;
char buffer[MAXBUF + ];
fd_set rfds;
struct timeval tv;
int retval, maxfd = -;
if(argc != )
{
printf("argv format errno, pls:\n\t\t%s IP port\n", argv[]);
exit();
}
//创建socket
if((sockfd = socket(AF_INET, SOCK_STREAM, )) < )
{
perror("Socket");
exit(EXIT_FAILURE);
}
bzero(&dest, sizeof(dest));
dest.sin_family = AF_INET;
dest.sin_port = htons(atoi(argv[])); //参数2为端口号
if(inet_aton(argv[], (struct in_addr *)&dest.sin_addr.s_addr) == ) //参数1为IP地址
{
perror(argv[]);
exit(EXIT_FAILURE);
}
//发起连接
if(connect(sockfd, (struct sockaddr *)&dest, sizeof(dest)) != )
{
perror("Connect");
exit(EXIT_FAILURE);
}
printf("\nget ready pls chat\n");
while()
{
FD_ZERO(&rfds);
FD_SET(, &rfds);
FD_SET(sockfd, &rfds);
maxfd = sockfd;
tv.tv_sec = ;
tv.tv_usec = ;
//多路复用
retval = select(maxfd + , &rfds, NULL, NULL, &tv);
if(retval == -)
{
printf("select %s", strerror(errno));
break;
}
else if(retval == ) //超时
continue;
else
{
if(FD_ISSET(sockfd, &rfds))
{
bzero(buffer, MAXBUF + );
len = recv(sockfd, buffer, MAXBUF, );
if(len > )
{
printf("recv message:'%s', %dbyte recv\n", buffer, len);
}
else if(len < )
{
printf("message recv failure\n");
}
else
{
printf("the other quit, quit\n");
break;
}
}
if(FD_ISSET(, &rfds))
{
bzero(buffer, MAXBUF + );
fgets(buffer, MAXBUF, stdin);
if(!strncasecmp(buffer, "quit", ))
{
printf("i will quit!\n");
break;
}
len = send(sockfd, buffer, strlen(buffer) - , );
if(len > )
printf("send successful, %d byte send!\n", len);
else
printf("send failure!");
}
}
}
close(sockfd);
return ;
}
服务器效果
客户端1效果
客户端2效果
【linux高级程序设计】(第十四章)TCP高级应用 2的更多相关文章
- 读书笔记 - js高级程序设计 - 第十五章 使用Canvas绘图
读书笔记 - js高级程序设计 - 第十三章 事件 canvas 具备绘图能力的2D上下文 及文本API 很多浏览器对WebGL的3D上下文支持还不够好 有时候即使浏览器支持,操作系统如果缺缺 ...
- linux高级管理第十四章--kvm虚拟化
案例 安装kvm所需软件 验证 注:虚拟机要开启虚拟引擎 开启服务 环境准备 安装相关软件包 启动 创建网桥 重启,reboot 安装虚拟机 完成.
- 第十四章:高级I/O
14.1:引言 本章内容包括非阻塞I/O.记录锁.系统V流机制.I/O多路转接(select和poll函数).readv和writev函数以及存储映射I/O(mmap),这些都称为高级I/O. 14. ...
- 鸟哥的Linux私房菜——第十四章:Bash Shell
视频链接:http://www.bilibili.com/video/av10094012/ 本章目录: 1. Bash shell1.1 什么是 shell ? (我们通过shell与Kernel核 ...
- 【TCP/IP详解 卷一:协议】第二十四章 TCP的未来与性能
来到了TCP的最后一个章节,未来与性能.在当时(1991年)的未来,如今已经部分变为现实,部分就只是历史中的实验. 主要内容: 路径MTU的发现与TCP的结合. 长肥管道 和 高速千兆比网络. 窗口扩 ...
- 【读书笔记】C#高级编程 第二十四章 文件和注册表操作
(一)文件和注册表 对于文件系统操作,相关的类几乎都在System.IO名称空间中,而注册表操作由System.Win32名称空间中的类来处理. (二)管理文件系统 System.MarshalByR ...
- 《javascript高级程序设计》第四章 Variables,scope,and memory
4.1 基本类型和引用类型的值 primitive and reference values 4.1.1 动态的属性 dynamic properties 4.1.2 复制变量值 copying va ...
- JavaScript高级程序设计:第四章
变量.作用域和内存问题 1.ECMAScript变量可能包含两种不同数据类型的值:基本类型值和引用类型值.基本类型值指的是简单的数据段,引用类型值指的是有多个值构成的对象. 2.动态的属性:定义一个基 ...
- 《JAVASCRIPT高级程序设计》第四章
javascript变量是松散类型,它只是在特定时间表示特定值的一个名字而已:变量的值以及类型,可以在脚本的生命周期内改变.变量的类型,分为基本类型和引用类型两种,具体介绍如下图所示: 执行环境是Ja ...
- 《JavaScript 高级程序设计》第四章:变量、作用域和内存问题
目录 变量的引用 执行环境及作用域 作用域链延长 块级作用域 垃圾回收机制 变量的引用 当一个变量保存了基本数据类型时,此时对于变量的操作(赋值,运算)就是操作这个基本数据的本身,就算是赋值操作,赋值 ...
随机推荐
- PHP.23-ThinkPHP框架的三种模型实例化-(D()方法与M()方法的区别)
三种模型实例化 原则上:每个数据表应对应一个模型类(Home/Model/GoodsModel.class.php --> 表tp_goods) 1.直接实例化 和实例化其他类库一样实例化模型类 ...
- 1,VMware与Centos系统安装
选择性 pc可以选择 -纯系统 Linux/windows -双系统 Windows+Linux -虚拟化技术 Windows+vmware workstation 服务器 -物理机纯系统 -物理机+ ...
- 调用startActivityForResult后直接调用onActivityResult
人员都知道,可以经由过程应用 startActivityForResult() 和 onActivityResult() 办法来传递或接管参数. 然而在"轻听"项目中,还没比及被调 ...
- Win7更换锁屏和开机画面
技术交流群:233513714 每次开机被Windows千年不变的开机画面和锁屏画面丑到的小伙伴们可以看过来,通过简单的几步就可以改掉系统默认的开机画面. 1.首先Windows+r键输入regedi ...
- laravel5.5任务调度
目录 1. 定义调度 1.1 使用Closure 1.2 Artisan 命令调度 1.3 队列任务调度 1.4 Shell 命令调度 1.5 调度频率设置 1.6 闭包测试限制 1.7 避免任务重复 ...
- 《Cracking the Coding Interview》——第3章:栈和队列——题目2
2014-03-18 05:08 题目:实现一个栈,除了能进行push和pop之外,还能在O(1)时间内返回栈中最小的元素. 解法:用另一个“最小栈”存放最小的元素,每当有不小于当前最小值的元素进栈时 ...
- 运用Pascal来破坏DLL的一个实例
运用Pascal来破坏DLL文件的一个实例 关于Pascal静态调用和动态的调用DLL的学习您可以看Delphi/Lazarus栏目. Uses Dos; {调用DOS库} Const Root='C ...
- 每天一个Linux命令(7):pwd命令
pwd命令以绝对路径的方式显示用户当前工作目录. 语法 pwd(选项) 选项 --help:显示帮助信息: --version:显示版本信息. 实例 [root@localhost ~]# pwd / ...
- Python 3基础教程8--if else、if elif else
本文介绍if else语句,不多说,直接看例子. if elif else语句
- python基础实践(四)
# -*- coding:utf-8 -*-# Author:sweeping-monkwhy = "为什么要组织列表?"print(why)Chicken_soup = &quo ...