APUE 学习之旅

[APUE] 文件 I/O 之 Read/Write/Lseek

2018-01-10  本文已影响0人  疯狂的攻城狮

前言

上一篇文章,主讲了 Open/Close 的 API ,以及简单的用法.

代码 Git 地址 SuzhenProjects/ApueProject

常用函数复习

Tips: 通过 man 2 <Command> 可以查询你系统的这些 API 档案

Read 函数 ssize_t read(int fildes, void *buf, size_t nbyte);

Write 函数 ssize_t write(int fildes, const void *buf, size_t nbyte);

Tips: readwrite 函数,两者形式上,基本类似,利用这点,可以很方便的进行记忆

Lseek 函数 off_t lseek(int fildes, off_t offset, int whence);

offset 参数具体意义,是根据 whence 来确定的.
lseek 如果作用在管道, FIFO,或者网络套接字上,会返回错误ESPIPE

实战 C++

我们设计的这段程序,需要先在程序运行目录创建一个test.txt 文件.随机写入一些数据,然后保存并关闭该文件.

//
// Created by suzhen on 05/01/2018.
//

#include <unistd.h>
#include <cstdio>
#include <fcntl.h>
#include <cstring>
#include <cerrno>

int main(int argc, char **argv) {
    static constexpr char TEST_FILE[]{"test.txt"};
    int test_file_fd = ::open(TEST_FILE, O_RDWR, 0644);
    if (test_file_fd < 0) {
        printf("%s 文件打开失败, %s\n", TEST_FILE, strerror(errno));
        return 1;
    }
    //注意,这段程序其实我们只能读取 0xFF-1 个c har
    constexpr size_t read_buf_size = 0xFF;
    char read_buf[read_buf_size]{'\0'};
    ssize_t read_sz = ::read(test_file_fd, (void *) read_buf, read_buf_size - 1);
    if (read_sz < 0) {
        printf("文件读取失败, %s\n", strerror(errno));
        ::close(test_file_fd);
        return 2;
    }
    printf("文件内容是 \n%s\n", read_buf);
    //追加一段内容,由于 read 的作用,当前读写游标已经移动到了末尾
    constexpr char append_string[]{"\n我直接追加到末尾!"};
    ssize_t write_sz = ::write(test_file_fd, append_string, strlen(append_string));
    if (write_sz < 0) {
        printf("文件写入失败, %s\n", strerror(errno));
        ::close(test_file_fd);
        return 3;
    }
    //seek到指定位置写,这里为了简单,直接 seek 到文件的开头的位置
    off_t seek_offset = lseek(test_file_fd, 0, SEEK_SET);
    if (seek_offset != 0) {
        printf("偏移到文件开始的位置,操作失败\n");
        ::close(test_file_fd);
        return 4;
    }
    printf("成功偏移到文件开始的位置\n");
    constexpr char insert_string[]{"我直接插入到头部!这会覆盖之前的内容"};
    ssize_t insert_sz = ::write(test_file_fd, insert_string, strlen(insert_string));
    if (insert_sz < 0 || insert_sz != strlen(insert_string)) {
        printf("在文件头部写入数据,操作失败\n");
        ::close(test_file_fd);
        return 5;
    }
    return 0;
}

重点

使用 read,write,close 这三个函数,仿佛就在操作数组一样,我们没法很轻松的在文件头插入一行数据,只能在末尾写入一段字符.
一旦在文件的头部进行 write文件头部的数据,直接会覆盖之前的内容,这也许不是我们想要的操作,如果你想要更加直观的操作一个文件,我们可以使用 fopen,fput... 等等专业面向文件的 API.

上一篇下一篇

猜你喜欢

热点阅读