[12]APUE:高级 I/O
一、分散聚离(向量) I/O
[a] readv / writev
#include <sys/uio.h>
ssize_t readv(int fd, const struct iovec *iov, int iovcnt)
ssize_t writev(int fd, const struct iovec *iov, int iovcnt)
/*成功返回已操作的字节数量,出错返回 -1*/
struct iovec {
void *iov_base; //缓冲区地址
size_t len;
}
- 用于针对多个不连续缓冲区同时写入写出的原子操作,减少了系统调用次数,通常提供比线性 I/O 更高的效率与性能
- 参数 iov 是指向 iovec 结构体的数组的指针,当其元素数量 <=8 时,系统通常会在栈上分配空间以优化性能
- 按数组元素順序依次读写
[b] 样例
#include <sys/uio.h>
#include <fcntl.h>
#include <stdio.h>
#define N0 10
#define N1 20
#define N2 30 int
main(void)
{
char *file = "/var/log/messages";
int fd, i;
char test0[N0], test1[N1], test2[N2];
struct iovec iov[]; fd = open(file, O_RDONLY);
if (- == fd)
{
perror("open");
} iov[].iov_base = test0;
iov[].iov_len = N0;
iov[].iov_base = test1;
iov[].iov_len = N1;
iov[].iov_base = test2;
iov[].iov_len = N2; i = readv(fd, iov, );
if (- == i)
{
perror("readv");
}
((char *)(iov[].iov_base))[N0 - ] = '\n';
((char *)(iov[].iov_base))[N1 - ] = '\n';
((char *)(iov[].iov_base))[N2 - ] = '\n';
i = writev(, iov, );
if (- == i)
{
perror("writev");
}
}
fh@Server ~/APUE$ ./a.out
Dec
: Server newsys
og[]: logfile turned over
二、存储影射(内存影射)
[a] mmap / munmap
#include <sys/mman.h>
void *mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset) //成功返回分配到的内存位置(指针),出错返回 MAP_FAILED
int munmap(void *addr, size_t len) //成功返回 0,出錯返回 -1
- mmap:
- addr 通常设置为 0,表示由系统自动分配内存
- len 指定要 map 的内存区域字节长度
- prot 指定影射内存区的权限,由 PROT_READ / PROT_WRITE / PROT_EXEC 按位 OR 賦值,不能与 open 调用的权限位或其它系统权限策略冲突
- flags 常用选项:MAP_SHARED / MAP_PRIVATE,前者对影射内存区的更改会写入文件,后者采用写时复制(COW)创建内存副本,不修改源文件
- fd 是与将要 map 的目标文件相关联的文件描述符
- offset 指定从目标文件 map 的起始位置(距离文件开头的偏移量),如指定为 100,意为从第 100 字节开始 map 长度为 len 字节的内容到 mmap 函数返回的指针所指向的内存区
- mmap 函数分配出的内存都是以页为单位对齐的,通常适用于分配较大的内存;malloc 函数的分配方法通常是小内存(<128KB)在堆上分配,大内存使用内存影射
- 运行时获取页大小:sysconf(_SC_PAGESIZE)
- 内存影射优点:
- 直接操作内存,不使用缓冲区,避免了多余的数据复制
- 不调用 read、write 等系统调用,无上下文切换的开销
- 多个进程 map 同一文件的同一区段的时候,数据会在所有进程间共享
- 文件偏移只需要移动指针,无须再调用 lseek
- 内存影射的缺点:
- 映射区域的大小总是页大小的整数倍,对于小文件会造成内存浪费
- 创建和维护映射及相关内核数据结构会有一定的开销,但相对于其节省的开销,是可以忽略的
[b] msync / mprotect
#include <sys/mman.h>
int msync(void *addr, size_t len, int flags) //成功返回 0,出错返回 -1
int mprotect(const void *addr, size_t len, int prot) //同上
- msync 相当于专门用于内存影射的 fsync,flags 可用选项有 MS_SYNC / MS_ASYNC,前者只有当数据真正已写回磁盘,msync 函数才返回,后者立即返回,数据何时真正写回磁盘,由系统调度
- mprotect 用于更改指定内存区域的权限,通常是先前用 mmap 分配的区域,但 Linux 可以用于更改任意用户空间可用的内存区域
三、POSIX I/O 提示
#include <fcntl.h>
int posix_fadvise(int fd, off_t offset, off_t len, int advice) //成功返回 0,出错不设置 errno
The behavior is specified by the advice parameter and may be one of: POSIX_FADV_NORMAL Tells the system to revert to the default data
access behavior. POSIX_FADV_RANDOM Is a hint that file data will be accessed
randomly, and prefetching is likely not
advantageous. POSIX_FADV_SEQUENTIAL Tells the system that file data will be accessed
sequentially. This currently does nothing as the
default behavior uses heuristics to detect
sequential behavior. POSIX_FADV_WILLNEED Tells the system that the specified data will be
accessed in the near future. The system may
initiate an asynchronous read of the data if it is
not already present in memory. POSIX_FADV_DONTNEED Tells the system that the specified data will not
be accessed in the near future. The system may
decrease the in-memory priority of clean data
within the specified range and future access to
this data may require a read operation. POSIX_FADV_NOREUSE Tells the system that the specified data will only
be accessed once and then not reused. The system
may decrease the in-memory priority of data once
it has been read or written. Future access to
this data may require a read operation.
- 指定恰当的预读策略,可大幅提升普通文件 I/O 性能
- offset 为 0 时,提示从文件开头预读,当 offset 与 len 同时为 0 时,表示整个文件
[12]APUE:高级 I/O的更多相关文章
- 12. javacript高级程序设计-DOM2和DOM3
1. DOM2和DOM3 DOM2级规范定义了一些模块,用于增强DOM1级.“DOM2级核心”为不同的DOM类型引入了一些与XML命名空间有关的方法,这些变化只在使用XML或者XHTML,对于HTML ...
- 12.mysql高级查询
1. mysql 支持三种类型的连接查询: on 后面跟的是关联条件 内连接查询 select s.name,c.name from students as s inner join classes ...
- 12.创建高级联结---SQL
一.使用表别名 SQL除了可以对列名和计算字段使用别名,还允许给表名起别名.这样做有两个主要理由: 缩短SQL语句: 允许在一条SELECT语句中多次使用相同的表. SELECT cust_name, ...
- linQ学习笔记之三高级语句
linq语句查询执行的时机 第一步获取数据源 int [] obejct = new int[]{1,2,3,4,5,6,7,8,9} 第二步定义查询 var even = numbers.where ...
- 软件测试人员必备Linux命令(初、中、高级)
有些技能可以事半功倍,有些命运掌握在我们手中.熟练的掌握和使用这些命令可以提高工作效率,并且结合这些命令对测试过程中遇到的问题进行一些初步的定位. 1 目录与文件操作1.1 ls(初级)使用权限:所有 ...
- 第32章 TIM—高级定时器—零死角玩转STM32-F429系列
第32章 TIM—高级定时器 全套200集视频教程和1000页PDF教程请到秉火论坛下载:www.firebbs.cn 野火视频教程优酷观看网址:http://i.youku.com/fire ...
- 高级程序员与CTO技术总监首席架构师
一.高级程序员 如果你是一个刚刚创业的公司,公司没有专职产品经理和项目经理,你就是公司的产品经理,你如果对你现在的开发员能力不满,那么你只需要的是一个高级程序员. 你定义功能.你做计划推进和管理,他可 ...
- SQL 必知必会
本文介绍基本的 SQL 语句,包括查询.过滤.排序.分组.联结.视图.插入数据.创建操纵表等.入门系列,不足颇多,望诸君指点. 注意本文某些例子只能在特定的DBMS中实现(有的已标明,有的未标明),不 ...
- [转]CTO、技术总监、首席架构师的区别
经常有创业公司老板来拜访我,常常会拜托给我一句话:帮我找一个CTO. 我解释的多了,所以想把这个写下来,看看你到底需要的应该是啥. 一.高级程序员 如果你是一个刚刚创业的公司,公司没有专职产品经理和项 ...
随机推荐
- NNVM打造模块化深度学习系统(转)
[摘录理由]: 之所以摘录本文,主要原因是:该文配有开源代码(https://github.com/dmlc/nnvm):读者能够直接体会文中所述的意义,便于立刻展开研究. MXNet专栏 :NNVM ...
- python---进程、线程
一.进程 二.线程 1,使用Thread类创建线程 #!/usr/bin/env python # -*- coding:utf-8 -*- __author__ = 'zhoufeng' impor ...
- php编码规范
PHP 文件格式 1.对于只包含有 PHP 代码的文件,结束标志("?>")是不允许存在的,PHP自身不需要("?>"), 这样做, 可以防止它的末 ...
- Go Data Structures: Interfaces
refer:http://research.swtch.com/interfaces Go Data Structures: Interfaces Posted on Tuesday, Decembe ...
- Android 在Canvas中实现画笔效果(一)--钢笔
如题: 公司要求做一个涂鸦板,要有钢笔.毛笔等画笔效果,网上搜了很多,可是效果不怎么好,决定自己研究下.废话不多说,进入正题. 首先,赛贝尔曲线弄明白了,在画曲线的过程中就是一条条的向量. 第二,曲线 ...
- 使用JSONObject遇到的问题,java.lang.NoClassDefFoundError: net/sf/json/JSONObject
先是报 java.lang.NoClassDefFoundError: net/sf/json/JSONObject 这个错误, 打开项目属性找到java build path中的libaries,找 ...
- jQuery:提交表单前判断表单是否被修改过
表单加载完成后执行 : //表单中包含input(text,checkbox,hidden),select,radio, $("#editWithdrawAutoApprovedConf ...
- Python常用模块学习
1.模块介绍 2.time & datetime模块 3.random 4.os 5.sys 6.shutil 7.json&pickle 8.shelve 9.xml处理 10.ya ...
- Redis基础(转)
ServiceStack.Redis实践 Redis的C#客户端我选择的是ServiceStack.Redis,相比Booksleeve redis-sharp等方案,它提供了一整套从 Redi ...
- 【Python全栈笔记】05 [模块二] 19 Oct 文件的操作
文件操作 一般步骤1. 文件打开 2. 文件操作 3. 文件关闭 1. 打开文件 使用open(文件名(绝对路径), 打开模式, 编码) 文件打开的模式有: r: 只读模式(默认) w: 只写模式 ...