open, creat - 用来 打开和创建 一个 文件或设备

SYNOPSIS 总览

#includ e <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int open(const char *pathname, int flags);
int open(const char *pathname, int flags, mode_t mode)
int creat(const char *pathname, mode_t mode);

描述 (DESCRIPTION)

open() 通常 用于 将 路径名 转换为 一个 文件描述符 (一个 非负的 小 整数, 在 read , write 等 I/O操作中 将会被使用). 当 open() 调用 成功, 它会 返回 一个 新的 文件描述符 (永远取 未用 描述符的 最小值). 这个调用 创建 一个 新的 打开文件, 即 分配 一个 新的 独一无二的 文件描述符, 不会与 运行中的 任何 其他程序 共享 (但 可以 通过 fork (2) 系统调用 实现 共享). 这个 新的 文件描述符 在其后 对 打开文件操作 的函数 中 使用.(参考 fcntl(2)). 文件的 读写 指针 被 置于 文件头

参数 flags 是通过 O_RDONLY, O_WRONLYO_RDWR (指明 文件 是以 只读 , 只写 或 读写 方式 打开的) 与 下面的 零个 或 多个 可选模式 按位 -or 操作 得到的:

O_CREAT
若文件 不存在 将 创建 一个 新 文件. 新 文件 的 属主 (用户ID) 被 设置 为 此 程序 的 有效 用户 的 ID. 同样 文件 所属 分组 也 被 设置 为 此 程序 的 有效 分组 的 ID 或者 上层 目录 的 分组 ID (这 依赖 文件系统 类型 ,装载选项 和 上层目录 的 模式, 参考,在 mount(8) 中 描述 的 ext2 文件系统 的 装载选项 bsdgroupssysvgroups )
O_EXCL
通过 O_CREAT, 生成 文件 , 若 文件 已经 存在 , 则 open 出错 , 调用 失败 . 若是 存在 符号联接 , 将会 把 它的 联接指针 的 指向 文件 忽略. O_EXCL is broken on NFS file systems, programs which rely on it for performing locking tasks will contain a race condition. The solution for performing atomic file locking using a lockfile is to create a unique file on the same fs (e.g., incorporating hostname and pid), use link(2) to make a link to the lockfile. If link() returns 0, the lock is successful. Otherwise, use stat(2) on the unique file to check if its link count has increased to 2, in which case the lock is also successful.
O_NOCTTY
假如 pathname 引用 一个 终端设备 --- 参考 tty(4) --- 即使 进程 没有 控制终端 ,这个 终端 也 不会 变成 进程 的 控制 终端.
O_TRUNC
假如 文件 已经 存在 , 且是 一个 普通 文件 ,打开 模式 又是 可写(即 文件 是 用 O_RDWR 或 O_WRONLY 模式 打开 的) , 就把 文件 的 长度 设置 为 零 , 丢弃 其中的 现有 内容.若 文件 是 一个 FIFO 或 终端设备 文件 , O_TRUNC 标志 被 忽略. 其他 O_TRUNC 的 作用 是 不 具体 指定 的 (在 许多 Linux 版本 中 , 通常 会 被 忽略 , 其他 的 一些 版本 将 返回 一个 错误)
O_APPEND
文件 以 追加 模式 打开 . 在 以前 , 文件 读写 指针 被 置 在 文件 的 末尾 . as if with lseek. O_APPEND may lead to corrupted files on NFS file systems if more than one process appends data to a file at once. This is because NFS does not support appending to a file, so the client kernel has to simulate it, which can't be done without a race condition.
O_NONBLOCKO_NDELAY
打开(open) 文件 可以 以 非块(non-blocking) 模式 打开 . 此时 文件 并 没有 打开 , 也 不能 使用 返回 的文件描述符 进行 后续 操作 , 而是 使 调用 程序 等待 . 此 模式 是 为了 FIFO (命名管道) 的 处理 , 参考 fifo(4). 这种 模式 对 除了 FIFO 外 没有 任何 影响 .
O_SYNC
打开 文件 实现 I/O 的 同步 . 任何 通过 文件描述符 对 文件 的 write 都会 使 调用 的 进程 中断 , 直到 数据 被 真正 写入 硬件 中 . 其他 , 参考 RESTRICTIONS.
O_NOFOLLOW
假如 pathname 是 一个 符号 联接 , 则 打开 失败 . 这是 FreeBSD 的 扩充 , 从 2.1.126 版本 以来 被 引入 到 Linux 中来 . 从 glibc2.0.100 库 以来 , 头文件 中 包括 了 这个 参数 的 定义;
kernel 2.1.126 以前 将 忽略 它的 使用.
O_DIRECTORY
假如 pathname 不是 目录 , 打开 就 失败 . 这个 参数 是 Linux 特有 的 , 在 kernel 2.1.126 中 加入 , 为了 避免 在 调用 FIFO 或 磁带设备 时 的 denial-of-service 问题 , 但是 不应该 在 执行 opendir 以外 使用.
O_LARGEFILE
在 32位 系统 中 支持 大 文件系统 , 允许 打开 那些 用 31位 都 不能 表示 其 长度 的 大 文件 .

在 文件 打开 后 , 这些 可选 参数 可以 通过 fcntl 来 改变 .

在 新文件 被 创建 时 , 参数 mode 具体 指明 了 使用 权限 . 他 通常 也 会 被 umask 修改 . 所以 一般 新建 文件 的 权限 为 (mode & ~umask). 注意 模式 只 被 应用 于 将来 对 这 新文件 的 使用 中; open 调用 创建 一个 新的 只读 文件 , 但 仍 将 返回 一个 可 读写 文件 描述符.

后面 是 一些 mode 的 具体 参数:

S_IRWXU
00700 允许 文件 的 属主 读 , 写 和 执行 文件
S_IRUSR (S_IREAD)
00400 允许 文件 的 属主 读 文件
S_IWUSR (S_IWRITE)
00200 允许 文件 的 属主 写 文件
S_IXUSR (S_IEXEC)
00100 允许 文件 的 属主 执行 文件
S_IRWXG
00070 允许 文件 所在 的 分组 读 , 写 和 执行 文件
S_IRGRP
00040 允许 文件 所在 的 分组 读 文件
S_IWGRP
00020 允许 文件 所在 的 分组 写 文件
S_IXGRP
00010 允许 文件 所在 的 分组 执行 文件
S_IRWXO
00007 允许 其他 用户 读 , 写 和 执行 文件
S_IROTH
00004 允许 其他 用户 读 文件
S_IWOTH
00002 允许 其他 用户 写 文件
S_IXOTH
00001 允许 其他 用户 执行 文件

mode 只有 当 在 flags 中 使用 O_CREAT 时 才 有效 , 否则 被 忽略.

creat 相当 于 open 的 参数 flags 等于 O_CREAT|O_WRONLY|O_TRUNC.

RETURN VALUE 返回值

opencreat 都 返回 一个 新的 文件描述符 (若是 有 错误 发生 返回 -1 ,并在 errno 设置 错误 信息). 注意 open 可以 打开 设备 专用 文件 , 但是 creat 不能创建,需要用 mknod(2) 来代替.

On NFS file systems with UID mapping enabled, open may return a file descriptor but e.g. read(2) requests are denied with EACCES. This is because the client performs open by checking the permissions, but UID mapping is performed by the server upon read and write requests.

若 文件 是 新 建立 的 , 他 的 atime(上次访问时间), ctime(创建时间), mtime(修改时间) 都 被 修改 为 当前 时间 , 上层 目录 的atime , ctime 也 被 同样 修改 . 其他的 , 假如 文件 是 由 O_TRUNC 参数 修改的 ,它的 ctime , mtime 域 也 被 设置 为 当前 时间.

 

ERRORS 错误信息

EEXIST
参数 O_CREAT and O_EXCL 被使用,但是文件( pathname )已经存在.
EISDIR
文件名 ( pathname ) 是 一个 目录 , 而 又 涉及 到 写 操作.
EACCES

访问 请求 不 允许 (权限不够) , 在 文件名 ( pathname )中 有 一 目录 不允许 搜索 (没有 执行权限) , 或者 文件 还 不存在 且 对 上层目录 的 写 操作 又 不允许.
ENAMETOOLONG
文件名 ( pathname ) 太 长 了
ENOENT
目录 ( pathname ) 不存在 或者 是 一个 悬空 的 符号 联接.
ENOTDIR
pathname 不是 一个 子目录
ENXIO
使用 O_NONBLOCK | O_WRONLY, 命名 的 文件 是 FIFO , 所读 文件 还 没有 打开 的 文件 , 或者 , 打开 一个 设备 专用 文件 而 相应 的 设备 不存在
ENODEV
文件 ( pathname ) 引用 了 一个 设备 专用 文件 , 而 相应 的 设备 又 不存在. (这是 linux kernel 的 一个bug - ENXIO 一定 会 被 返回 .)
EROFS
文件 ( pathname ) 是 一个 只读 文件 , 又有 写 操作 被 请求.
ETXTBSY
文件 ( pathname ) 是 一个 正在 被 执行 的 可 执行 文件 ,又有 写 操作 被 请求.
EFAULT
pathname 在一个你不能访问的地址空间.
ELOOP
在 分解 pathname 时 , 遇到 太多 符号联接 或者 指明 O_NOFOLLOW 但是 pathname 是 一个 符号联接
ENOSPC
pathname 将要被创建,但是设备又没有空间储存 pathname 文件了
ENOMEM
可 获得 的 核心内存(kernel memory) 不够
EMFILE
程序打开的文件数已经达到最大值了
ENFILE
系统打开的总文件数已经达到了极限 

转载于:https://www.cnblogs.com/fanweisheng/p/11097350.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
这样算计算读写速度吗?#include <stdio.h> #include <string.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include <sys/time.h> int main(int argc, char *argv[]) { char filename[] = "aFile"; // 即将创建打开文件名 char *bufForReadWrite = (char *)malloc(1024 * 1024 * 100); // 申请100MB空间作为读写缓冲区 size_t bufSize = sizeof(bufForReadWrite); //测量出读写缓冲区大小 int fd; //用于存放open函数返回的文件描述符 struct timeval tv_start, tv_end; // 用于存放开始和结束时间 // 打开创建文件并写入数据 fd = open(filename, O_RDWR | O_CREAT, 0644); // 创建打开文件,可读可写,创建权限为644 if(fd < 0) { // 创建文件失败时返回 perror("Unable to create file"); return -1; } gettimeofday(&tv_start, NULL); // 记录写入开始时间 ssize_t sizeWritten = write(fd, (void *)bufForReadWrite, bufSize); printf("Write: %ld bytes to %s\n", sizeWritten, filename); close(fd); // 写入完毕后关闭文件 // 打开文件并读取数据 fd = open(filename, O_RDWR); // 打开文件后的权限为可读可写 if(fd < 0) { //打开文件失败时返回 perror("Unable to open file"); return -1; } ssize_t sizeRead = read(fd, (void *)bufForReadWrite, bufSize); gettimeofday(&tv_end, NULL); // 记录读取结束时间 printf("Read: %ld bytes from %s, read content:%s\n", sizeRead, filename, bufForReadWrite); close(fd); // 读完毕后关闭文件 // 计算读写速度 double time = (tv_end.tv_sec - tv_start.tv_sec) * 1000.0; // 将时间转为豪秒 time += (tv_end.tv_usec - tv_start.tv_usec) / 1000.0; // 加上微秒部分的时间 double speed__write_read = (sizeWritten / (1024.0 * 1024.0)) / (time / 1000.0); // 读写速度 printf("Read speed: %.2f MB/s\n", speed_write_read); return 0; }
06-01

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值