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设置为非阻塞方式 :

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

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

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

示例代码:

  1. #include <sys/types.h>
  2. #include <sys/socket.h>
  3. #include <sys/wait.h>
  4. #include <stdio.h>
  5. #include <stdlib.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. #include <sys/un.h>
  9. #include <sys/time.h>
  10. #include <sys/ioctl.h>
  11. #include <unistd.h>
  12. #include <netinet/in.h>
  13. #include <fcntl.h>
  14. #include <unistd.h>
  15. #define SERVPORT 3333
  16. #define BACKLOG 10
  17. #define MAX_CONNECTED_NO 10
  18. #define MAXDATASIZE 100
  19.  
  20. int main()
  21. {
  22. struct sockaddr_in server_sockaddr,client_sockaddr;
  23. int sin_size,recvbytes,flags;
  24. int sockfd,client_fd;
  25. char buf[MAXDATASIZE];
  26. /*创建socket*/
  27. if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1){
  28. perror("socket");
  29. exit(1);
  30. }
  31. printf("socket success!,sockfd=%d\n",sockfd);
  32.  
  33. /*设置sockaddr结构*/
  34. server_sockaddr.sin_family=AF_INET;
  35. server_sockaddr.sin_port=htons(SERVPORT);
  36. server_sockaddr.sin_addr.s_addr=INADDR_ANY;
  37. bzero(&(server_sockaddr.sin_zero),8);
  38.  
  39. /*将本地ip地址绑定端口号*/
  40. if(bind(sockfd,(struct sockaddr *)&server_sockaddr,sizeof(struct sockaddr))==-1){
  41. perror("bind");
  42. exit(1);
  43. }
  44. printf("bind success!\n");
  45.  
  46. /*监听*/
  47. if(listen(sockfd,BACKLOG)==-1){
  48. perror("listen");
  49. exit(1);
  50. }
  51. printf("listening....\n");
  52.  
  53. /*fcntl()函数,处理多路复用I/O*/
  54. if((flags=fcntl( sockfd, F_GETFL, 0))<0)
  55. perror("fcntl F_GETFL");
  56. flags |= O_NONBLOCK;
  57. if(fcntl( sockfd, F_SETFL,flags)<0)
  58. perror("fcntl");
  59. while(1){
  60. sin_size=sizeof(struct sockaddr_in);
  61. if((client_fd=accept(sockfd,(struct sockaddr*)&client_sockaddr,&sin_size))==-1){ //服务器接受客户端的请求,返回一个新的文件描述符
  62. perror("accept");
  63. exit(1);
  64. }
  65. if((recvbytes=recv(client_fd,buf,MAXDATASIZE,0))==-1){
  66. perror("recv");
  67. exit(1);
  68. }
  69. if(read(client_fd,buf,MAXDATASIZE)<0){
  70. perror("read");
  71. exit(1);
  72. }
  73. printf("received a connection :%s",buf);
  74.  
  75. /*关闭连接*/
  76. close(client_fd);
  77. close(sockfd);
  78. exit(0);
  79. }/*while*/
  80. }

运行结果:

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

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

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

  1. if((flags=fcntl( sockfd, F_SETFL, 0))<0)
  2. perror("fcntl F_SETFL");
  3. flags |= O_ASYNC;
  4. if(fcntl( sockfd, F_SETFL,flags)<0)
  5. perror("fcntl");

运行结果如下:

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

可以看到,进程一直处于等待中,直到另一相关信号驱动它为止。

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

  1. #include <sys/types.h>
  2. #include <sys/socket.h>
  3. #include <sys/wait.h>
  4. #include <stdio.h>
  5. #include <stdlib.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. #include <sys/un.h>
  9. #include <sys/time.h>
  10. #include <sys/ioctl.h>
  11. #include <unistd.h>
  12. #include <netinet/in.h>
  13. #define SERVPORT 3333
  14. #define BACKLOG 10
  15. #define MAXDATASIZE 100
  16. int main()
  17. {
  18. struct sockaddr_in server_sockaddr,client_sockaddr;
  19. int sin_size,recvbytes;
  20. fd_set readfd;
  21. fd_set writefd;
  22. int sockfd,client_fd;
  23. char buf[MAXDATASIZE];
  24. /*创建socket*/
  25. if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1){
  26. perror("socket");
  27. exit(1);
  28. }
  29. printf("socket success!,sockfd=%d\n",sockfd);
  30. /*设置sockaddr结构*/
  31. server_sockaddr.sin_family=AF_INET;
  32. server_sockaddr.sin_port=htons(SERVPORT);
  33. server_sockaddr.sin_addr.s_addr=INADDR_ANY;
  34. bzero(&(server_sockaddr.sin_zero),8);
  35. /*将本地ip地址绑定端口号*/
  36. if(bind(sockfd,(struct sockaddr *)&server_sockaddr,sizeof(struct sockaddr))==-1){
  37. perror("bind");
  38. exit(1);
  39. }
  40. printf("bind success!\n");
  41. /*监听*/
  42. if(listen(sockfd,BACKLOG)==-1){
  43. perror("listen");
  44. exit(1);
  45. }
  46. printf("listening....\n");
  47. /*select*/
  48. FD_ZERO(&readfd); // 将readfd 清空
  49. FD_SET(sockfd,&readfd); //将sockfd加入到readfd集合中
  50. while(1){
  51. sin_size=sizeof(struct sockaddr_in);
  52. int max_fd = sockfd + 1;
  53. if(select(max_fd,&readfd,NULL,NULL,(struct timeval *)0)>0){ //第一个参数是0和sockfd的最大值加1,第二个参数是读集,第三、四个参数是写集 //和异常集
  54. if(FD_ISSET(sockfd,&readfd)>0){ // FD_ISSET 这个宏判断 sockfd 是否属于可读的文件描述符。从 sockfd 中读入, 输出到标准输出上去.
  55. if((client_fd=accept(sockfd,(struct sockaddr *)&client_sockaddr,&sin_size))==-1){ //client_sockaddr:客户端地址
  56. perror("accept");
  57. exit(1);
  58. }
  59. if((recvbytes=recv(client_fd,buf,MAXDATASIZE,0))==-1){
  60. perror("recv");
  61. exit(1);
  62. }
  63. if(read(client_fd,buf,MAXDATASIZE)<0){
  64. perror("read");
  65. exit(1);
  66. }
  67. printf("received a connection :%s",buf);
  68. }
  69. close(client_fd);
  70. close(sockfd);
  71. }/*select*/
  72. }/*while*/
  73. }

运行结果:

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

UNIX网络编程——fcntl函数的更多相关文章

  1. UNIX网络编程——select函数的并发限制和 poll 函数应用举例

    一.用select实现的并发服务器,能达到的并发数,受两方面限制 1.一个进程能打开的最大文件描述符限制.这可以通过调整内核参数.可以通过ulimit -n来调整或者使用setrlimit函数设置,  ...

  2. UNIX网络编程——ioctl 函数的用法详解

    1.介绍 Linux网络程序与内核交互的方法是通过ioctl来实现的,ioctl与网络协议栈进行交互,可得到网络接口的信息,网卡设备的映射属性和配置网络接口.并且还能够查看,修改,删除ARP高速缓存的 ...

  3. UNIX网络编程——sockatmark函数

    每当收到一个带外数据时,就有一个与之关联的带外标记.这是发送进程发送带外字节时该字节在发送端普通数据流中的位置.在从套接字读入期间,接收进程通过调用sockatmark函数确定是否处于带外标记. #i ...

  4. UNIX网络编程——客户/服务器心搏函数

    阅读此博客时,可以参考以前的博客<<UNIX网络编程--socket的keep-alive>>和<<UNIX网络编程--套接字选项(心跳检测.绑定地址复用)> ...

  5. UNIX网络编程——getsockname和getpeername函数

    UNIX网络编程--getsockname和getpeername函数   来源:网络转载   http://www.educity.cn/linux/1241293.html     这两个函数或者 ...

  6. UNIX网络编程——UDP 的connect函数(改进版)

    上一篇我们提到,除非套接字已连接,否则异步错误是不会返回到UDP套接字的.我们确实可以给UDP套接字调用connect,然而这样做的结果却与TCP连接大相径庭:没有三次握手.内核只是检查是否存在立即可 ...

  7. 【unix网络编程第三版】阅读笔记(五):I/O复用:select和poll函数

    本博文主要针对UNP一书中的第六章内容来聊聊I/O复用技术以及其在网络编程中的实现 1. I/O复用技术 I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备就绪,它就通知该进程.I/O ...

  8. UNIX网络编程——使用select函数编写客户端和服务器

    首先看原先<UNIX网络编程--并发服务器(TCP)>的代码,服务器代码serv.c: #include<stdio.h> #include<sys/types.h> ...

  9. UNIX网络编程 第6章 I/O复用:select和poll函数

    UNIX网络编程 第6章 I/O复用:select和poll函数

随机推荐

  1. Linux学习之CentOS(十八)-----恢复Ext3下被删除的文件与 使用grep恢复被删文件内容(转)

    前言 下面是这个教程将教你如何在Ext3的文件系统中恢复被rm掉的文件. 删除文件 假设我们有一个文件名叫 'test.txt' $ls -il test.txt 15 -rw-rw-r– 2 roo ...

  2. WEB中间件--Jboss未授权访问,

    1,Jboss未授权访问部署木马 发现存在Jboss默认页面,点进控制页 点击 Jboss.deployment 进入应用部署页面 也可以直接输入此URL进入 http://www.ctfswiki. ...

  3. 正则替换replace中$1的用法以及常用正则

    一.repalce定义 用于在字符串中用一些字符替换另一些字符,或替换一个与正则表达式匹配的子串. stringObject.replace(regexp/substr,replacement)参数一 ...

  4. fastDFS 安装 配置 使用

    fastDFS 安装 配置 使用 关于安装 本文采用的是源码的安装方式,其他安装方式请自行百度 简单介绍 1.背景 FastDFS是一款开源的.分布式文件系统(Distributed File Sys ...

  5. jQuery 选择器 prop() 和attr()

    Day30 jQuery 1.1.1.1 什么是jQuery? n jQuery是javaScript的前端框架.对常见的对象和常用的方法进行封装,使用更方便. 它兼容CSS3,还兼容各种浏览器.文档 ...

  6. python学习之路前端-HTML

    HTML概述 HTML是英文Hyper Text Mark-up Language(超文本标记语言)的缩写,他是一种制作万维网页面标准语言(标记).相当于定义统一的一套规则,大家都来遵守他,这样就可以 ...

  7. CSharpGL(48)用ShadowVolume画模型的影子

    CSharpGL(48)用ShadowVolume画模型的影子 在Per-Fragment Operations & Tests阶段,有一个步骤是模版测试(Stencil Test).依靠这一 ...

  8. Java面试06|项目相关介绍

    1.明确你的项目到底是做什么的,有哪些功能 广告投放机:项目主要是为移动端有针对性的进行广告展示. 媒体管理平台SSP:为媒体端实现多种变现途径 (1)广告投放机中关于广告检索与排序的功能 1.广告检 ...

  9. 焦点轮播图(tab轮播)

    主要有两部分:1.列表导航(小图片) 2.展示区(大图片) 页面布局: HTML部分:    <div class="s_conC">                  ...

  10. beta 分布的详细介绍(转载)

    目前看到的关于beta分布最好的一个解释,由于贴过来格式不好看,所以附上链接: http://www.datalearner.com/blog/1051505532393058