Linux系统文件编程

作为大家熟悉的文件管理功能我就不多说了,这里主要提一些浅显的文件操作。
首先提到的是系统调用,什么是系统调用呢?所谓系统调用,是指操作系统提供给用户程序调用的一组“特殊”接口,用户程序可以通过这组“特殊”接口来获得操作系统内核提供的服务。因为安全的原因用户程序不能直接访问系统内核提供的服务,由于在Linux中,为了更好地保护内核空间,将程序的运行空间分为内核空间和用户空间(也就是常称的内核态和用户态),它们分别运行在不同的级别上,在逻辑上是相互隔离的。因此,用户进程在通常情况下不允许访问内核数据,也无法使用内核函数,它们只能在用户空间操作用户数据,调用用户空间的函数。
Linux中,一切都是文件,在Linux中对目录和设备的操作都等同于对文件的操作,都是使用文件描述符来进行的。Linux文件可分为:普通文件,目录文件,链接文件,设备文件。
先来介绍一下系统调用下的创建文件。
int creat(const char *filename, mode_t mode )
filename :创建的文件名
(包含路径,缺省为当前路径)
mode:创建模式
常创建模式:
S_IRUSR 可读
S_IWUSR 可写
S_IXUSR 可执行
S_IXRWU 可读、可写、可执行
除用以上宏来选择创建模式,也可以用数字来表示,creat基本上很少用了,因为open这个函数也有creat的功能,open一个没有的文件时会自动创建这个文件,使用open函数时需要指定路径,打开方式,打开方式有三种:O_RDONLY 只读打开 , O_WRONLY 只写打开 , O_RDWR 读、写打开,必须指定一种方式且只有一种。关闭一个文件就很简单了,直接close(路径)就行。接下来介绍读和写。
read:
int read(int fd, const void *buf, size_t length)
功能就是从文件描述符fd所指定的文件中读取length个字节到buf所指向的缓冲区中,返回值为实际读取的字节数。

write:
int write(int fd, const void * buf,  size_t length)

功能就是把length个字节从buf指向的缓冲区中写到文件描述符fd所指向的文件中,返回值为实际写入的字节数。

然后就是定位,lseek,int lseek(int fd, offset_t offset, int whence)
这是一个文件指针,相当于写的最后一个位置的下一位,功能就是将文件读写指针相对whence移动offset个字节。操作成功时,返回文件指针相对于文件头的位置。whence可以用宏函数来写,SEEK_SET表示相对文件开头,SEEK_CUR表示当前位置,SEEK_END表示文件的末尾。offset可取负值,表示向前移动。例如下述调用可将文件指针相对当前位置向前移动5个字节:lseek(fd, -5, SEEK_CUR)。
基本上系统调用常用的函数都说了,这里写一个代码小结一下。
我们就往一个文件里写一个结构体吧。

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <unistd.h>

struct node
{
    char name[20];
    char home[20];
    char phone[11];
};

typedef struct node Node;

int main()
{
    Node n = {David, Nanjing, 13345678923};
    Node tmp;
    int fd, ret, i;

    fd = open("Mynote.txt", O_CREAT | O_RDWR | O_EXCL, 755);
    if(-1 == fd)
    {
        perror("open");
        exit(1);
    }            //在写入之前要打开文件

    ret = write(fd, &n, sizeof(Node));
    if(-1 == ret)
    {
        perror("write");
        exit(1);
    }           //这里写入一个结构体到文件中

    printf("%d \n", ret); //这里打印的是写入的结构体长度

    ret = lseek(fd, 0, SEEK_SET);
    if(-1 == ret)
    {
        perror("lseek");
        exit(1);
    }           //写入之后我们来读一下这个文件,所以要把指针移到最前面,不然接着读只能是空的

    ret = read(fd, &tmp, sizeof(Node));
    if(-1 == ret)
    {
        perror("read");
        exit(1);
    }      

    printf("%d \n", ret); //这里打印的是阅读的结构体长度

    close(fd);           //最后别忘了关闭文件
    return 0;
}

接下来聊聊标准库函数,C库函数的文件操作是独立于具体的操作系统平台的,不管是在DOS、Windows、Linux还是在VxWorks中都是这些函数。标准I/O提供缓存的目的就是减少调用read和write的次数,它对每个I/O流自动进行缓存管理(标准I/O函数通常调用malloc来分配缓存)。它提供了三种类型的缓存:
1) 全缓存。当填满标准I/O缓存后才执行I/O操作。磁盘上的文件通常是全缓存的。
2) 行缓存。当输入输出遇到新行符或缓存满时,才由标准I/O库执行实际I/O操作。stdin、stdout通常是行缓存的。
3) 无缓存。相当于read、write了。stderr通常是无缓存的,因为它必须尽快输出。
一般而言,由系统选择缓存的长度,并自动分配。标准I/O库在关闭流的时候自动释放缓存。
在标准I/O库中,一个不足之处是需要复制的数据量。当每次使用函数fgets和fputs时,通常需要复制两次数据:
第一次是在内核和标准I/O缓存之间(当调用read和write时),
第二次是在标准I/O缓存(通常系统分配和管理)和用户程序中的行缓存(fgets的参数就需要一个用户行缓存指针)之间。
FILE *fopen(const char *filename, const char *mode)
filename:打开的文件名(包含路径,缺省为当前路径)
mode: 打开模式
r, rb : 只读方式打开,文件必须已存在
w,wb : 只写方式打开,如果文件不存在则创建,
如果文 件已存在清空重写
a, ab: 只能在文件末尾追加数据,如果文件不存在则创建
r+,rb+,r+b: 读写方式打开,文件必须已存在
w+,w+b,wb+: 读写方式打开,如果文件不存在则创建,如果文件已存在清空重写
a+,a+b,ab+: 读和追加方式打开,如果文件不存在则创建
b用于区分二进制文件和文本文件,这一点在DOS、Windows系统中是有区分的,但Linux不区分二进制文件和文本文件。
fgets( 由文件中读取一字符串),表头文件:include

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main()
{
    FILE *fp;
    size_t ret;
    char buf[100] = {0};

    scanf("%s", buf);

    fp = fopen("Hello.txt","w");
    if(NULL == fp)
    {
        perror("fopen");
        exit(1);
    }

    ret = fwrite(buf, 1, strlen(buf), fp);
    if(0 == ret)
    {
        perror("fwrite");
        exit(1);
    }                        //把从终端输入的字符串写到Hello.txt(不存在则创建)中

    fclose(fp); 

    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值