UNIX网络编程——fcntl函数

fcntl函数提供了与网络编程相关的如下特性:

  • 非阻塞式I/O。  通过使用F_SETFL命令设置O_NONBLOCK文件状态标志,我们可以把一个套接字设置为非阻塞型。
  • 信号驱动式I/O。 通过使用F_SETFL命令设置O_ASYNC文件状态标志,我们可以把一个套接字设置成O_ASYNC,一旦其状态发生变化,内核就产生一个SIGIO信号。
  • F_SETOWN命令允许我们指定用于接收SIGIO和SIGURG信号的套接字属主(进程ID或进程组ID)。其中SIGIO信号是套接字被设置为信号驱动式I/O型产生的,SIGURG信号是在新的带外数据到达套接字时产生的。F_GETOWN命令返回套接字的当前属主。

 

fcntl()函数有如下特性:

  • 非阻塞I/O: 可将cmd 设为F_SETFL,将lock设为O_NONBLOCK。
  • 信号驱动I/O:可将cmd设为F_SETFL,将lock设为O_ASYNC。


用以下方法将socket设置为非阻塞方式 :

int flags = fcntl(socket, F_GETFL, 0); 
fcntl(socket, F_SETFL, flags | O_NONBLOCK);

将非阻塞的设置回阻塞可以用:

int flags = fcntl(socket, F_GETFL, 0); 
fcntl(socket, F_SETFL, flags & ~O_NONBLOCK);


示例代码:

#include <sys/types.h>  
#include <sys/socket.h>  
#include <sys/wait.h>  
#include <stdio.h>  
#include <stdlib.h>  
#include <errno.h>  
#include <string.h>  
#include <sys/un.h>  
#include <sys/time.h>  
#include <sys/ioctl.h>  
#include <unistd.h>  
#include <netinet/in.h>  
#include <fcntl.h>  
#include <unistd.h>  
#define SERVPORT 3333  
#define BACKLOG 10  
#define MAX_CONNECTED_NO 10  
#define MAXDATASIZE 100  int main()  
{  struct sockaddr_in server_sockaddr,client_sockaddr;  int sin_size,recvbytes,flags;  int sockfd,client_fd;  char buf[MAXDATASIZE];  
/*创建socket*/  if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1){  perror("socket");  exit(1);  }  printf("socket success!,sockfd=%d\n",sockfd);  /*设置sockaddr结构*/  server_sockaddr.sin_family=AF_INET;  server_sockaddr.sin_port=htons(SERVPORT);  server_sockaddr.sin_addr.s_addr=INADDR_ANY;  bzero(&(server_sockaddr.sin_zero),8);  /*将本地ip地址绑定端口号*/  if(bind(sockfd,(struct sockaddr *)&server_sockaddr,sizeof(struct sockaddr))==-1){  perror("bind");  exit(1);  }  printf("bind success!\n");  /*监听*/  if(listen(sockfd,BACKLOG)==-1){  perror("listen");  exit(1);  }  printf("listening....\n");  /*fcntl()函数,处理多路复用I/O*/  if((flags=fcntl( sockfd, F_GETFL, 0))<0)  perror("fcntl F_GETFL");  flags |= O_NONBLOCK;  if(fcntl( sockfd, F_SETFL,flags)<0)  perror("fcntl");  while(1){  sin_size=sizeof(struct sockaddr_in);  if((client_fd=accept(sockfd,(struct sockaddr*)&client_sockaddr,&sin_size))==-1){  //服务器接受客户端的请求,返回一个新的文件描述符  perror("accept");  exit(1);  }  if((recvbytes=recv(client_fd,buf,MAXDATASIZE,0))==-1){  perror("recv");  exit(1);  }  if(read(client_fd,buf,MAXDATASIZE)<0){  perror("read");  exit(1);  }  printf("received a connection :%s",buf);  /*关闭连接*/  close(client_fd); close(sockfd);	exit(0);  }/*while*/  
}  
运行结果:

huangcheng@ubuntu:~$ ./a.out
socket success!,sockfd=3
bind success!
listening....
accept: Resource temporarily unavailable

可以看到,当accept的资源不可用时,程序会自动返回。

若将54--58行代码替换为:

if((flags=fcntl( sockfd, F_SETFL, 0))<0)  perror("fcntl F_SETFL");  
flags |= O_ASYNC; 
if(fcntl( sockfd, F_SETFL,flags)<0)  perror("fcntl");  

运行结果如下:

huangcheng@ubuntu:~$ ./a.out
socket success!,sockfd=3
bind success!
listening....
可以看到,进程一直处于等待中,直到另一相关信号驱动它为止。


由select函数实现的,示例代码:

#include <sys/types.h>  
#include <sys/socket.h>  
#include <sys/wait.h>  
#include <stdio.h>  
#include <stdlib.h>  
#include <errno.h>  
#include <string.h>  
#include <sys/un.h>  
#include <sys/time.h>  
#include <sys/ioctl.h>  
#include <unistd.h>  
#include <netinet/in.h>  
#define SERVPORT 3333  
#define BACKLOG 10   
#define MAXDATASIZE 100  
int main()  
{  struct sockaddr_in server_sockaddr,client_sockaddr;  int sin_size,recvbytes;  fd_set readfd;  fd_set writefd;  int sockfd,client_fd;  char buf[MAXDATASIZE];  
/*创建socket*/  if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1){  perror("socket");  exit(1);  }  printf("socket success!,sockfd=%d\n",sockfd);  
/*设置sockaddr结构*/  server_sockaddr.sin_family=AF_INET;  server_sockaddr.sin_port=htons(SERVPORT);  server_sockaddr.sin_addr.s_addr=INADDR_ANY;  bzero(&(server_sockaddr.sin_zero),8);  
/*将本地ip地址绑定端口号*/  if(bind(sockfd,(struct sockaddr *)&server_sockaddr,sizeof(struct sockaddr))==-1){  perror("bind");  exit(1);  }  printf("bind success!\n");  
/*监听*/  if(listen(sockfd,BACKLOG)==-1){  perror("listen");  exit(1);  }  printf("listening....\n");  
/*select*/  FD_ZERO(&readfd);              // 将readfd 清空   FD_SET(sockfd,&readfd);         //将sockfd加入到readfd集合中  while(1){  sin_size=sizeof(struct sockaddr_in);int max_fd = sockfd + 1;	if(select(max_fd,&readfd,NULL,NULL,(struct timeval *)0)>0){  //第一个参数是0和sockfd的最大值加1,第二个参数是读集,第三、四个参数是写集                                                                                //和异常集  if(FD_ISSET(sockfd,&readfd)>0){         // FD_ISSET 这个宏判断 sockfd 是否属于可读的文件描述符。从 sockfd 中读入, 输出到标准输出上去.  if((client_fd=accept(sockfd,(struct sockaddr *)&client_sockaddr,&sin_size))==-1){   //client_sockaddr:客户端地址  perror("accept");  exit(1);  }  if((recvbytes=recv(client_fd,buf,MAXDATASIZE,0))==-1){  perror("recv");  exit(1);  }  if(read(client_fd,buf,MAXDATASIZE)<0){  perror("read");  exit(1);  }  printf("received a connection :%s",buf);  } close(client_fd); close(sockfd);		}/*select*/  }/*while*/  
}  
运行结果:

huangcheng@ubuntu:~$ ./a.out
socket success!,sockfd=3
bind success!
listening....

转载于:https://www.cnblogs.com/wangfengju/p/6172570.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/408623.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何远程访问服务器的 Jupyter notebook

图来自 Unsplash 网站&#xff0c;作者&#xff1a;Christopher Gower2019 年第 52 篇文章&#xff0c;总第 76 篇文章本文大约 4600 字&#xff0c;阅读大约需要 12 分钟写在前面当我们拥有一台服务器的时候&#xff0c;通常服务器都可能包含比本地电脑比较好的配置&#xff0…

[Python技巧]如何加快循环操作和Numpy数组运算速度

2019 年第 53 篇文章&#xff0c;总第 77 篇文章本文大约 4200 字&#xff0c;阅读大约需要 11 分钟前言Python 虽然写起来代码量要远少于如 C,Java&#xff0c;但运行速度又不如它们&#xff0c;因此也有了各种提升 Python 速度的方法技巧&#xff0c;这次要介绍的是用 Numba …

ictclas4j 分词工具包 安装流程

首先把 ictclasj解压缩&#xff0c;然后 1.把 Data文件夹整个拷贝到 Eclipse项目的文件夹下, 2.而 bin目录下的 org文件夹整个拷贝到你 Eclipse项目的 bin目录下&#xff0c;&#xff08;将class文件存进去&#xff09; 3.把src目录下的org文件夹整个拷贝到 Eclipse项目 的src目…

[周末阅读]认知和规划,以及推荐几个入门教程Github

图片来自 Unsplash&#xff0c;作者&#xff1a; Rafael Saes 2019 年第 54 篇文章&#xff0c;总第 78 篇文章本文大约 4600 字&#xff0c;阅读大约需要 12 分钟前言本来预计每个周末一篇读后感&#xff0c;但从上次分享[周末阅读]如何培养你的自信和正确认识财富、创造财富&…

Android开发(一):android环境搭建

android开发环境搭建&#xff0c;图文并茂&#xff0c;推荐http://jingyan.baidu.com/article/f0062228f0b18afbd2f0c871.html 【SDK Manager.exe】安装过程遇到问题&#xff1a;【Android SDK Manager下载API时一直显示“Done loading packages”却迟迟不能前进......】 解决地…

Python基础入门_5面向对象基础

Python 基础入门前四篇&#xff1a; Python 基础入门–简介和环境配置Python基础入门_2基础语法和变量类型Python基础入门_3条件语句和迭代循环Python基础入门_4函数 第五篇主要介绍 Python 的面向对象基础知识&#xff0c;也就是类的介绍&#xff0c;包括类方法和属性、构造…

ios 制作framework

原文&#xff1a;http://db-in.com/blog/2011/07/universal-framework-iphone-ios-2-0/ 原文废话太多啊&#xff0c;自己总结一下&#xff0c;因为我是在原有的静态库工程基础上建立的&#xff0c;所以新建一个target就好了。 1 新建target&#xff0c;macOS中的bundle&#xf…

一文了解下 GANs可以做到的事情

原文链接&#xff1a;https://machinelearningmastery.com/impressive-applications-of-generative-adversarial-networks/ 简介 如果说目前深度学习最火&#xff0c;应用最多的领域&#xff0c;莫过于 GAN–Generative Adversarial Network&#xff0c;翻译过来就是生成对抗…

浅析php学习的路线图

一直都想走上码农的道路&#xff0c;奈何当年没有学好。一直与码农无缘。现在又想做一些码农就下了一套某个培训机构的php视频来看。希望能走上码农的道路1.php初级教程 初级教程主要的页面设置的&#xff0c;就是 htmljsdivcss 2.中级教程中级的话开始接触php,就是php核心编程…

被垃圾分类逼疯?试试这款垃圾自动分类器

这是小编转载的第 31 篇好文来自&#xff1a;视说AI作者&#xff1a;视说君让垃圾自动分类近期垃圾分类成为了一个热门话题&#xff0c;原来直接一次性扔掉的垃圾&#xff0c;现在都需要分门别类进行投放。从今年7月1日起&#xff0c;新的《上海市生活垃圾管理条例》正式开始施…

Python基础入门6_文件和异常

Python 基础入门前五篇&#xff1a; Python 基础入门–简介和环境配置Python基础入门_2基础语法和变量类型Python基础入门_3条件语句和迭代循环Python基础入门_4函数Python基础入门_5面向对象基础 这次将介绍有关文件和异常的处理&#xff0c;包括读写文本文件、二进制文件、…

VSS Get Latest Version 没有提示recursive的对话框解决

今天按照VSS使用时&#xff0c;当“Get Latest version”时&#xff0c;不小心勾选了 “Only show this dialog when the Shift key is down”&#xff0c;因此当我再“Get Latest version”&#xff0c;该对话框不会出来&#xff0c;所以造成没发获取全部code。 解决办法&…