linux select 网络模型
io模型:
同步IO: 阻塞形式,非阻塞形式(轮询)、信号驱动IO、IO复用(select, poll, epoll);
异步io:aio_read()
典型场景:
1、客户端处理多种IO-------标准io 和网络io(套接字)
2、server端既要处理监听套接字又要处理已连接的套接字
3、一个服务器要处理TCP,又要处理UDP
4、server端要处理多种服务多个协议;
5、不管是网络套接字还是文件读写描述符均可处理;
#include <stdio.h>
#include <sys/select.h>
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h> #include <netinet/in.h>
#include <sys/socket.h>
#include <string.h> //bzero() int main()
{
int socketfd;
struct sockaddr_in serv_addr; //struct socket address internet
int i =;
/** /usr/include/netinet/in.h
* #define __SOCKADDR_COMMON(sa_prefix) \
* sa_family_t sa_prefix##family)
*
* struct sockaddr_in{
* _SOCKADDR_COMMON(sin_); //展开宏sa_family_t sin_family
* int_port_t sin_port;
* struct in_addr sin_addr;
* }
*accept默认会阻塞进程,直到有一个客户连接建立后返回,
*它返回的是一个新可用的套接字,这个套接字是连接套接字。
*此时我们需要区分两种套接字,一种套接字正如accept的参数sockfd,它是监听套接字,
*在调用listen函数之后,一个套接字会从主动连接的套接字变身为一个监听套接字;
*而accept返回是一个连接套接字,它代表着一个网络已经存在的点点连接。
*在调用listen函数之后,一个套接字会从主动连接的套接字变身为一个监听套接字;
*而accept返回是一个连接套接字,它代表着一个网络已经存在的点点连接。
*自然要问的是:为什么要有两种套接字?原因很简单,如果使用一个描述字的话,
*那么它的功能太多,使得使用很不直观,同时在内核确实产生了一个这样的新的描述字.
*/
//struct sockaddr_in serv_addr;
//address family, socket_stream, 0:会自动选择type类型对应的默认协议
socketfd = socket(AF_INET,SOCK_STREAM, );
if(socketfd){ } bzero((char*)&serv_addr, sizeof(serv_addr));
serv_addr.sin_family = AF_INET; //主机字节序
serv_addr.sin_port = htons(); //网络字节序
serv_addr.sin_addr.s_addr =htonl(INADDR_ANY); //监听所有地址 if(bind(socketfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr))){ } listen(socketfd, ); int maxfd;
fd_set allset, rset;
maxfd = socketfd; FD_ZERO(&allset); //将allset清零,使得集合中不含任何fd
FD_SET(socketfd, &allset); //将socketfd加入allset集合 int sockfd, clilen, bytes;
struct sockaddr_in cli_addr;
char buffer[]; while(){
rset = allset;
/*select第一个参数表示要检查的文件描述符的个数;比如最大值为5,因为
*文件描述符是从0开始的,所以需要检查0,1,2,3,4,5这六个值;即最大值+1;
*/
if(select(maxfd+, &rset, NULL, NULL, NULL)){ } for(i=; i<=maxfd; i++){
if(FD_ISSET(i, &rset)){ //一个个来判断是否被置位了
if(socketfd == i){ //监听套接字
clilen = sizeof(cli_addr);
sockfd = accept(socketfd, (struct sockaddr*)&cli_addr, (socklen_t*)&clilen);
if(sockfd<){ //error
perror("accept error \n");
}
FD_CLR(i, &rset); //清零
maxfd = maxfd>sockfd?maxfd:sockfd;
FD_SET(sockfd, &allset); }else{ //通信套接字
bzero(buffer, );
bytes = recv(i, buffer, , );
if(bytes <= ){ //client端退出
FD_CLR(i, &allset); //清零
close(i);
continue;
}
printf("recv msg: %s", buffer);
send(i, buffer, , );
}
}
}
} return ;
}
linux select 网络模型的更多相关文章
- linux select 与 阻塞( blocking ) 及非阻塞 (non blocking)实现io多路复用的示例
除了自己实现之外,还有个c语言写的基于事件的开源网络库:libevent http://www.cnblogs.com/Anker/p/3265058.html 最简单的select示例: #incl ...
- linux—select具体解释
linux—select具体解释 select系统调用时用来让我们的程序监视多个文件句柄的状态变化的.程序会停在select这里等待,直到被监视的文件句柄有一个或多个发生了状态改变. 关于文件句柄,事 ...
- Linux Select之坑
最近在写一个demo程序,调用select()来监听socket状态,流程如下: r_set 初始化 timeout 初始化3秒超时 loop{ select(ntfs, &r_set, nu ...
- linux select 与 阻塞( blocking ) 及非阻塞 (non blocking)实现io多路复用的示例【转】
转自:https://www.cnblogs.com/welhzh/p/4950341.html 除了自己实现之外,还有个c语言写的基于事件的开源网络库:libevent http://www.cnb ...
- linux select函数详解
linux select函数详解 在Linux中,我们可以使用select函数实现I/O端口的复用,传递给 select函数的参数会告诉内核: •我们所关心的文件描述符 •对每个描述符,我们所关心的状 ...
- Linux select 机制深入分析
Linux select 机制深入分析 作为IO复用的实现方式.select是提高了抽象和batch处理的级别,不是传统方式那样堵塞在真正IO读写的系统调用上.而是堵塞在sele ...
- Linux select TCP并发服务器与客户端编程
介绍:运行在ubuntu linux系统,需要先打开一个终端运行服务端代码,这时,可以打开多个终端同时运行多个客户端代码(注意客户端数目要小于MAX_FD);在客户端输入数据后回车,可以看见服务器收到 ...
- Linux select I/O 复用
用途 在处理多个socket套接字的时候,会很自然的遇到一个问题:某个套接字什么时候可读?什么时候可写?哪些套接字是需要关闭的?我们可以回忆一下,一般我们在最开始编写socket程序的时候,send, ...
- select网络模型知识总结
select模型支持IO多路复用,select函数如下 int select ( IN int nfds, //windows下无意义,linux有意义 IN OUT fd_set* readfds, ...
随机推荐
- [原创]linux简单之美(一)
原文链接:linux简单之美(一) 话说windows也有syscall,这是必须的.但是win的syscall可以直接call吗?可以是可以但是破费周折,搞成SDT之类的复杂概念.下面看看linux ...
- 解决Oracle 11gR2 空闲连接过多,导致连接数满的问题
今天又遇到了11gR2连接数满的问题,以前也遇到过,因为应用那边没有深入检查,没有找到具体原因,暂且认为是这个版本Oracle的BUG吧. 上次的处理办法是用Shell脚本定时在系统中kill v$ ...
- J2SE知识点摘记(二十)
List 1.3.1 概述 前面我们讲述的Collection接口实际上并没有直接的实现类.而List是容器的一种,表示列表的意思.当我们不知道存储的数据有多少的情况,我们就可以使用Li ...
- 分析Ext2文件系统结构。
1. 目的 分析Ext2文件系统结构. 使用 debugfs 应该跟容易分析 Ext2文件系统结构 了解ext2的hole的 2. 准备工作 预习文件系统基本知识: http://www.doc88. ...
- 起启航-华夏互联与杰华网络合体结盟打造本土IT利舰
北京时间2013年9月9日消息: 领先的软件研发企业上海逐一软件科技有限公司与专业互联网推广运营机构南昌杰华网络开发有限公司达成协议,双方将建立紧密合作关系与集团运营体制,并在未来的10个月内进行相应 ...
- C++四种强制类型转换详解
什么是类型转换? 类型转换的含义是通过改变一个变量的类型为别的类型从而改变该变量的表示方式.为了类型转换一个简单对象为另一个对象你会使用传统的类型转换操作符. C与C++的类型转换 //C中: //复 ...
- .Net 数组去除重复项
string str = "1/1/12/13/15/16/15//"; ] { '/' }, StringSplitOptions.RemoveEmptyEntries); Li ...
- pycharm中添加扩展工具pylint
今天调试了好几个小时,想吧pylint集成到pycharm中去,从网上找了个宝贝帖 子,但是不好用,原因是作者写的脚本是检查工程和模块的,而我的是单独检查一个文件,当然前者肯定会在项目后期用的.所以就 ...
- AppDelegate关于应用程序挂起、复原与终止的代理方法
AppDelegate关于应用程序挂起.复原与终止的代理方法: 首次运行: - (BOOL)application:(UIApplication *)application didFinishLaun ...
- 详解new/delete(整合)
C++中内存的动态分配与管理永远是一个让C++开发者头痛的问题,本文通过对C++中内存的动态分配释放的基本原理的介绍,让读者朋友能对C++中的内存的动态分配与释放有较为深入的理解,从而更好驾驭C++程 ...