底层文件访问:
    进程:运行中的程序,它有一些与值关联的文件描述符,有多少个文件描述符取决于系统配置情况。
    当一个程序开始运行时,一般会打开三个文件描述符:
        0:标准输入
        1:标准输出
        2:标准错误
        可以通过系统调用(open)把其他文件描述符与文件和设备相关联。
       
    write系统调用:
        作用:把缓冲区(buf)的前nbytes个字节写入与文件描述符(fildes)关联的文件。
        write返回实际写入的字节数,如果文件描述符有错误或者底层设备的驱动程序对数据长度比较敏感,表示在write调用中出现了
        错误,返回值可能会小于nbytes。如果函数返回0,表示未写入任何数据;返回-1表示write调用中出现了错误,错误代码保存在
        全局变量errno中。

    #include <stdio.h>
    #include <unistd.h>    // write调用原型所在的头文件
    int main(void)
    {
      if ( (write(, "Here is some data\n", )) != )
        write(, "A write error has occurred on file descriptor 1\n", );
      return ;
    }

        root@ubuntu:/home/linlin/linlin/c_code# ./simple_write.out
        Here is some data

write可能会报告写入的字节比要求的少,这不一定时错误,在程序中,需要检查errno以发现错误,然后再次调用write写入剩
    余数据
    
  read系统调用:
      作用:从文件描述符(fildes)相关联的文件里读入nbytes个字节的数据,并把它们放到数据区buf中。
      read返回实际读入的字节数,这可能会小于请求的字节数,如果read调用返回0,表示未读入任何数据,已到达了文件尾;如果返
      回-1,表示read调用出现了错误。

    #include <stdio.h>
    #include <unistd.h>
    int main(void)
    {
      char buffer[];
      int nread;
      nread = read(, buffer, );
      if (- == nread)
        write(, "A read error has occurred\n", );
      if ( (write(, buffer, nread)) != nread )
        write(, "A write error has occurred\n", );
      return ;
    }

root@ubuntu:/home/linlin/linlin/c_code# echo hello there | ./simple_read.out
        hello there
       
    open调用:
        作用:创建一个新的文件描述符。

        #include <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);

open建立了一条到文件或设备的访问路径,如果调用成功,它可以返回一个可以被read,write和其它系统调用使用的文件描述符
        这个文件描述符时唯一的,它不会与任何其他运行中的进程共享。
        准备打开的文件或设备的名字作为参数path传递给函数,oflags参数用于指定打开文件所采取的动作。
        oflags参数是通过必须文件访问模式与其它可选模式相结合的方式指定的:
            模式                       说明
            O_RDONLY                   以只读方式打开
            O_WRONLY                   以只写方式打开
            O_RDWR                     以读写方式打开
       
        open调用还可以在oflags参数中包含可选模式的组合(用按位或操作):
            O_APPEND: 把写入数据追加在文件末尾
            O_TRUNC:把文件长度设置为0,丢弃已有的内容
            O_CREAT: 如果需要,就按照参数mode中给出的访问模式创建文件
            O_EXCL:与O_CREAT一起使用,确保调用者创建出文件。open调用是一个原子操作,它只执行一个函数调用。使用这个可选模
            式可以防止两个程序同时创建同一个文件。如果文件已存在,open调用将失败。
           
   
    访问权限的初始值:
        当使用代用O_CREAT标志的open调用来创建文件时,必须使用带有三个参数格式的open调用,第三个参数mode是几个标志按位或得
        到的,这些标志在sys/stat.h中定义:
            S_IRUSR: 读权限,文件属主
            S_IWUSR:写权限,文件属主
            S_IXUSR:执行权限,文件属主
            S_IRGRP: 读权限,文件属组
            S_IWGRP:写权限,文件属组
            S_IXGRP:执行权限,文件属组
            S_IROTH: 读权限,其他用户
            S_IWOTH:写权限,其他用户
            S_IXOTH:执行权限,其他用户

    #include <stdio.h>
    #include <fcntl.h>
    #include <sys/types.h>
    #include <sys/stat.h>
    int main(void)
    {
      open("test", O_CREAT, S_IRUSR|S_IXOTH);
      return ;
    }

文件访问权限的影响因素:
        指定的文件权限只有在创建文件时才会使用
        umask(用户掩码)会影响到被创建文件的访问权限,open调用中给出的mode值将与当时的用户掩码的反值做AND操作
        例如umask为001,open调用指定了S_IXOTH模式标志,那么创建的文件对其他用户不会拥有写执行权限。
       
    open和creat调用中的标志实际上是发出设置文件访问权限的请求,所请求的权限是否会被设置取决于当时的umask值。
    
    复制文件的两个不同实现:

    #include <unistd.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    int main(void)
    {
      char block[];
      int in, out;
      int nread;
      in = open("file.in", O_RDONLY);
      out = open("file.m.out", O_WRONLY|O_CREAT, S_IRUSR|S_IWUSR);
      while ((nread = read(in, block, sizeof(block))) > )
        write(out, block, nread);
      return ;
    }
    #include <unistd.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    int main(void)
    {
      char c;
      int in, out;
      in = open("file.in", O_RDONLY);
      out = open("file.out", O_WRONLY|O_CREAT, S_IRUSR|S_IWUSR);
      while (read(in, &c, ) == )
        write(out, &c, );
      return ;
    }

close系统调用:
        使用close调用终止文件描述符fildes与其对应文件之间的关联。文件描述符能被释放并能够重新使用。
        close调用成功时返回0,出错时返回-1

 #include <unistd.h>
       
int close(int fildes);

检查close调用的返回结果非常重要,有的文件系统,特别是网络文件系统,可能不会再关闭文件之前报告文件写操作中
        出现的错误,这是因为在执行操作时数据可能未被正确写入。
       
   
    ioctl系统调用:
        作用;提供一个控制设备及其描述符行为和配置底层服务的接口。
        终端,文件描述符,套接字设置磁带机都可以有为它们定义的ioctl。

  #include <unistd.h>
       
int ioctl(int fildes, int cmd, ...);

ioctl对描述符fildes引用的对象执行cmd参数中给出的操作。根据特定设备支持操作的不同,它可能会有一个可选的第三参数。

Linux——系统调用笔记1的更多相关文章

  1. Linux内核笔记--内存管理之用户态进程内存分配

    内核版本:linux-2.6.11 Linux在加载一个可执行程序的时候做了种种复杂的工作,内存分配是其中非常重要的一环,作为一个linux程序员必然会想要知道这个过程到底是怎么样的,内核源码会告诉你 ...

  2. Linux 读书笔记 二

    一.实验说明 1. 环境登录 无需密码自动登录,系统用户名shiyanlou,密码shiyanlou 若不小心登出后,直接刷新页面即可 2. 环境使用 完成实验后可以点击桌面上方的“实验截图”保存并分 ...

  3. linux内核剖析(六)Linux系统调用详解(实现机制分析)

    本文介绍了系统调用的一些实现细节.首先分析了系统调用的意义,它们与库函数和应用程序接口(API)有怎样的关系.然后,我们考察了Linux内核如何实现系统调用,以及执行系统调用的连锁反应:陷入内核,传递 ...

  4. Linux 学习笔记之超详细基础linux命令 Part 14

    Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122 ---------------------------------接Part 13---------------- ...

  5. # linux读书笔记(3章)

    linux读书笔记(3章) 标签(空格分隔): 20135328陈都 第三章 进程管理 3.1 进程 进程就是处于执行期的程序(目标码存放在某种存储介质上).但进程并不仅仅局限于一段可执行程序代码( ...

  6. linux读书笔记(5章)

    linux读书笔记(5章) 标签(空格分隔): 20135328陈都 第五章 系统调用 5.1 与内核通信 系统调用 让应用程序受限的访问硬件设备 提供创建新进程并与已有进程通信的机制 提供申请操作系 ...

  7. Linux学习笔记(七) 查询系统

    1.查看命令 (1)man 可以使用 man 命令名称 命令查看某个命令的详细用法,其显示的内容如下: NAME:命令名称 SYNOPSIS:语法 DESCRIPTION:说明 OPTIONS:选项 ...

  8. Linux学习笔记——管道PIPE

    管道:当从一个进程连接数据流到另一个进程时,使用术语管道(pipe).# include <unistd.h> int pipe(int filedes[2]); //创建管道 pipe( ...

  9. Linux 学习笔记

    Linux学习笔记 请切换web视图查看,表格比较大,方法:视图>>web板式视图 博客园不能粘贴图片吗 http://wenku.baidu.com/view/bda1c3067fd53 ...

随机推荐

  1. 串的基本操作(KMP算法实现)

    #include <iostream> #include <math.h> using namespace std; void StrAssign(char *T) { cha ...

  2. Git Bash 常用指令

    1. 关于git bash常用指令 推荐博客: 史上最简单的 GitHub 教程  猴子都能懂的GIT入门 Learn Version Control with Git for Free Git Do ...

  3. SQL语句操作SQL SERVER数据库登录名、用户及权限

    要想成功访问 SQL Server 数据库中的数据, 我们需要两个方面的授权: 获得准许连接 SQL Server 服务器的权利: 获得访问特定数据库中数据的权利(select, update, de ...

  4. 用Jenkins构建项目实战

    登录Jenkins,新建任务 输入一个任务名称,选择一个项目类型 使用自定义工作空间:使该项目独立于系统的工作空间 自动从Git下载源码,点击添加可以增加凭证 日程表的参数: 第一个参数代表的是分钟 ...

  5. ASP.NET MVC如何在页面加载完成后ajax异步刷新

    背景:之前已写过两篇有关Ajax的随笔,这一篇是单独针对在页面加载完成的Ajax操作.比如说打开学生列表页面,先加载页面,然后以Ajax的方式,从数据库中检索相应的学生信息,给浏览者更好的体验. 简单 ...

  6. loadrunner使用随机值

    用户登录设置:系统用1000000001.1000000002等可以登录系统,这个代表登录的用户名

  7. HDU 5421 Victor and String

    Victor and String Time Limit: 1000ms Memory Limit: 262144KB This problem will be judged on HDU. Orig ...

  8. 【分块+树状数组】codechef November Challenge 2014 .Chef and Churu

    https://www.codechef.com/problems/FNCS [题意] [思路] 把n个函数分成√n块,预处理出每块中各个点(n个)被块中函数(√n个)覆盖的次数 查询时求前缀和,对于 ...

  9. 关于NSArray的去重

    提到去重,第一反应就是for或while循环来遍历处理. 然后有了几种思路: 1) 两个循环嵌套 对比 然后去重: 但是这种方法会数据丢失, arr为要去重的数组 ; i<arr.count; ...

  10. linux下rename用法--批量重命名 转

    原文地址:https://www.cnblogs.com/hester/p/5615871.html Linux的rename 命令有两个版本,一个是C语言版本的,一个是Perl语言版本的,早期的Li ...