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....
UNIX网络编程——fcntl函数的更多相关文章
- UNIX网络编程——select函数的并发限制和 poll 函数应用举例
一.用select实现的并发服务器,能达到的并发数,受两方面限制 1.一个进程能打开的最大文件描述符限制.这可以通过调整内核参数.可以通过ulimit -n来调整或者使用setrlimit函数设置, ...
- UNIX网络编程——ioctl 函数的用法详解
1.介绍 Linux网络程序与内核交互的方法是通过ioctl来实现的,ioctl与网络协议栈进行交互,可得到网络接口的信息,网卡设备的映射属性和配置网络接口.并且还能够查看,修改,删除ARP高速缓存的 ...
- UNIX网络编程——sockatmark函数
每当收到一个带外数据时,就有一个与之关联的带外标记.这是发送进程发送带外字节时该字节在发送端普通数据流中的位置.在从套接字读入期间,接收进程通过调用sockatmark函数确定是否处于带外标记. #i ...
- UNIX网络编程——客户/服务器心搏函数
阅读此博客时,可以参考以前的博客<<UNIX网络编程--socket的keep-alive>>和<<UNIX网络编程--套接字选项(心跳检测.绑定地址复用)> ...
- UNIX网络编程——getsockname和getpeername函数
UNIX网络编程--getsockname和getpeername函数 来源:网络转载 http://www.educity.cn/linux/1241293.html 这两个函数或者 ...
- UNIX网络编程——UDP 的connect函数(改进版)
上一篇我们提到,除非套接字已连接,否则异步错误是不会返回到UDP套接字的.我们确实可以给UDP套接字调用connect,然而这样做的结果却与TCP连接大相径庭:没有三次握手.内核只是检查是否存在立即可 ...
- 【unix网络编程第三版】阅读笔记(五):I/O复用:select和poll函数
本博文主要针对UNP一书中的第六章内容来聊聊I/O复用技术以及其在网络编程中的实现 1. I/O复用技术 I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备就绪,它就通知该进程.I/O ...
- UNIX网络编程——使用select函数编写客户端和服务器
首先看原先<UNIX网络编程--并发服务器(TCP)>的代码,服务器代码serv.c: #include<stdio.h> #include<sys/types.h> ...
- UNIX网络编程 第6章 I/O复用:select和poll函数
UNIX网络编程 第6章 I/O复用:select和poll函数
随机推荐
- Linux学习之CentOS(十八)-----恢复Ext3下被删除的文件与 使用grep恢复被删文件内容(转)
前言 下面是这个教程将教你如何在Ext3的文件系统中恢复被rm掉的文件. 删除文件 假设我们有一个文件名叫 'test.txt' $ls -il test.txt 15 -rw-rw-r– 2 roo ...
- WEB中间件--Jboss未授权访问,
1,Jboss未授权访问部署木马 发现存在Jboss默认页面,点进控制页 点击 Jboss.deployment 进入应用部署页面 也可以直接输入此URL进入 http://www.ctfswiki. ...
- 正则替换replace中$1的用法以及常用正则
一.repalce定义 用于在字符串中用一些字符替换另一些字符,或替换一个与正则表达式匹配的子串. stringObject.replace(regexp/substr,replacement)参数一 ...
- fastDFS 安装 配置 使用
fastDFS 安装 配置 使用 关于安装 本文采用的是源码的安装方式,其他安装方式请自行百度 简单介绍 1.背景 FastDFS是一款开源的.分布式文件系统(Distributed File Sys ...
- jQuery 选择器 prop() 和attr()
Day30 jQuery 1.1.1.1 什么是jQuery? n jQuery是javaScript的前端框架.对常见的对象和常用的方法进行封装,使用更方便. 它兼容CSS3,还兼容各种浏览器.文档 ...
- python学习之路前端-HTML
HTML概述 HTML是英文Hyper Text Mark-up Language(超文本标记语言)的缩写,他是一种制作万维网页面标准语言(标记).相当于定义统一的一套规则,大家都来遵守他,这样就可以 ...
- CSharpGL(48)用ShadowVolume画模型的影子
CSharpGL(48)用ShadowVolume画模型的影子 在Per-Fragment Operations & Tests阶段,有一个步骤是模版测试(Stencil Test).依靠这一 ...
- Java面试06|项目相关介绍
1.明确你的项目到底是做什么的,有哪些功能 广告投放机:项目主要是为移动端有针对性的进行广告展示. 媒体管理平台SSP:为媒体端实现多种变现途径 (1)广告投放机中关于广告检索与排序的功能 1.广告检 ...
- 焦点轮播图(tab轮播)
主要有两部分:1.列表导航(小图片) 2.展示区(大图片) 页面布局: HTML部分: <div class="s_conC"> ...
- beta 分布的详细介绍(转载)
目前看到的关于beta分布最好的一个解释,由于贴过来格式不好看,所以附上链接: http://www.datalearner.com/blog/1051505532393058