linux c 笔记 文件(二)

1.dup和dup2函数

#include<unistd.h>
int dup(int fd);
int dup2(int fd1,int fd2);
两个均为复制一个现存的文件的描述
两个函数的返回:若成功为新的文件描述,若出错为-1;
由dup返回的新文件描述符一定是当前可用文件描述中的最小数值。用dup2则可以用fd2参数指定新的描述符数值。如果fd2已经打开,则先关闭。若fd1=fd2,则dup2返回fd2,而不关闭它。通常使用这两个系统调用来重定向一个打开的文件描述符。

复制一个描述符的另一种方法是使用 fcntl函数,下一节将对该函数进行说明。实际上,
调用:
dup (filedes) ;
等效于:fcntl (filedes, F_DUPFD, 0);
而调用:
dup2 (filedes ,filedes ,2);
等效于:
close (filedes2 ) ;
fcntl(filedes, F_DUPFD, filedes2);
在最后一种情况下,dup2并不完全等同于close加上fcntl。它们之间的区别是:
(1) dup2是一个原子操作,而 close及fcntl则包括两个函数调用。有可能在 close和 fcntl之间
插入执行信号捕获函数,它可能修改文件描述符。
(2) 在dup2和fcntl 之间有某些不同的errno。

2.根据文件描述词来操作文件的特性。fcntl

#include <unistd.h>
#include <fcntl.h>
int fcntl(int fd, int cmd);
int fcntl(int fd, int cmd, long arg);
int fcntl(int fd, int cmd, struct flock *lock);

fcntl()针对(文件)描述符提供控制。参数fd是被参数cmd操作(如下面的描述)的描述符。针对cmd的值,fcntl能够接受第三个参数int arg。
fcntl()的返回值与命令有关。如果出错,所有命令都返回-1,如果成功则返回某个其他值。下列三个命令有特定返回值:F_DUPFD , F_GETFD , F_GETFL以及F_GETOWN。
 
fcntl函数有五种功能:
• 复制一个现存的描述符( cmd =F_DUPFD)。
• 获得/设置文件描述符标记( cmd = F_GETFD或F_SETFD)。
• 获得/设置文件状态标志( cmd = F_GETFL或F_SETFL)。
• 获得/设置异步I / O有权( cmd = F_GETOWN或F_SETOWN)。
• 获得/设置记录锁( cmd = F_GETLK , F_SETLK或F_SETLKW)。  
    
1. cmd值的F_DUPFD :
    F_DUPFD    返回一个如下描述的(文件)描述符:
        ·最小的大于或等于arg的一个可用的描述符
        ·与原始操作符一样的某对象的引用
        ·如果对象是文件(file)的话,则返回一个新的描述符,这个描述符与arg共享相同的偏移量(offset)
        ·相同的访问模式(读,写或读/写)
        ·相同的文件状态标志(如:两个文件描述符共享相同的状态标志)
        ·与新的文件描述符结合在一起的close-on-exec标志被设置成交叉式访问execve(2)的系统调用

2. cmd值的F_GETFD和F_SETFD:      
    F_GETFD    取得与文件描述符fd联合的close-on-exec标志,类似FD_CLOEXEC。如果返回值和FD_CLOEXEC进行与运算结果是0的话,文件保持交叉式访问exec(),否则如果通过exec运行的话,文件将被关闭(arg 被忽略)        
    F_SETFD    设置close-on-exec标志,该标志以参数arg的FD_CLOEXEC位决定,应当了解很多现存的涉及文件描述符标志的程序并不使用常数 FD_CLOEXEC,而是将此标志设置为0(系统默认,在exec时不关闭)或1(在exec时关闭)

在修改文件描述符标志或文件状态标志时必须谨慎,先要取得现在的标志值,然后按照希望修改它,最后设置新标志值。不能只是执行F_SETFD或F_SETFL命令,这样会关闭以前设置的标志位。

3. cmd值的F_GETFL和F_SETFL:   
    F_GETFL    取得fd的文件状态标志,如同下面的描述一样(arg被忽略),在说明open函数时,已说明
了文件状态标志。不幸的是,三个存取方式标志 (O_RDONLY , O_WRONLY , 以及O_RDWR)并不各占1位。(这三种标志的值各是0 , 1和2,由于历史原因,这三种值互斥 — 一个文件只能有这三种值之一。) 因此首先必须用屏蔽字O_ACCMODE相与取得存取方式位,然后将结果与这三种值相比较。       
    F_SETFL    设置给arg描述符状态标志,可以更改的几个标志是:O_APPEND,O_NONBLOCK,O_SYNC 和 O_ASYNC。
而fcntl的文件状态标志总共有7个:O_RDONLY , O_WRONLY , O_RDWR , O_APPEND , O_NONBLOCK , O_SYNC和O_ASYNC
   

可更改的标志如下:
    O_NONBLOCK   非阻塞I/O,如果read(2)调用没有可读取的数据,或者如果write(2)操作将阻塞,则read或write调用将返回-1和EAGAIN错误
    O_APPEND     强制每次写(write)操作都添加在文件大的末尾,相当于open(2)的O_APPEND标志
    O_ASYNC      当I/O可用的时候,允许SIGIO信号发送到进程组,例如:当有数据可以读的时候

记录锁分为读取锁和写入锁,其中读取锁又称为共享锁,可以使多个
 * 进程都能够在文件的同一部分建立读取锁。而写入锁又称为互斥锁,
 * 在任何时刻只能有一个进程在文件的某个部分建立写入锁。当然,在
 * 文件的同一部分不能同时建立读取锁和写入锁。

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

//自定义错误处理函数
void my_err(const char * err_string , int line)
{
    fprintf (stderr ,"line :%d",line);
    perror (err_string);
    exit(1);
}

//锁的设置或者是释放
int lock_set (int fd ,struct flock * lock)
{
    if (fcntl (fd ,F_SETLK ,lock ) == 0 )    {/*执行成功*/
        if ( lock-> l_type ==F_RDLCK ){
            printf("set read lock , pid :%d\n  ",getpid() );
        }else if ( lock -> l_type ==F_WRLCK){
            printf("set write lock ,pid :%d \n ",getpid() );
        }else if ( lock -> l_type == F_UNLCK) {
            printf("release lock ,pid :%d \n",getpid());
        }
    }
    else {/*执行失败,返回-1*/
        perror("lock operation fail \n");
        return -1;
    }
    return 0;
}

//测试锁,只有当测试发现参数lock指定的锁能被设置时 ,返回 0

int lock_test(int fd, struct flock *lock )
{
    if(fcntl (fd, F_GETLK ,lock ) == 0 ){//执行成功
        if( lock->l_type == F_UNLCK){ //测试发现能按照参数lock要求设置锁}
            printf(" lock can be set  in fd \n");
            return 0;
        }else {//又不兼容的锁存在,打印出设置该锁的进程ID
        if( lock -> l_type ==F_RDLCK ){
            printf("can‘t  set lock ,read lock has set by: %d \n",lock -> l_pid );
        }else if (lock ->l_type ==F_WRLCK){
            printf("can‘t  set lock ,write lock has set by :%d \n ",lock ->l_pid);
        }
        return -2;
        }
    }else {
        perror ("get incompatible locks fail ");
    return -1;
    }

}

int main()
{
    int fd ;
    int ret;
    struct flock  lock;
    char  read_buf[32];
    //打开或者创建文件夹
    if ((fd = open ("example_65 ",O_CREAT | O_TRUNC |O_RDWR ,S_IRWXU)) == -1) {
        my_err ("open ",__LINE__);
    }
    if(write (fd ,"test lock ",10 ) !=10 ){
        my_err ("test ",__LINE__ );
    }
    //初始化lock结构
    memset (&lock , 0 ,sizeof (struct flock) );
    lock.l_start = SEEK_SET;
    lock.l_whence = 0;
    lock.l_len =0;
    
    //设置读锁
    lock.l_type =F_RDLCK;
     if ( lock_test (fd , &lock ) == 0) {//测试可以使用文件锁
        lock.l_type =F_RDLCK ;
        lock_set (fd ,&lock );
    }
    //读数据
    lseek (fd ,0 , SEEK_SET );
    if((ret = read (fd , read_buf ,10 )) < 0){
        my_err ("read ", __LINE__);
    }
    read_buf[ret] == ‘\0‘;
    printf ( "%s \n ",read_buf );
    //等待任意按键
    getchar();

//设置写锁
    lock.l_type = F_WRLCK ;
    if (lock_test(fd ,&lock ) == 0) {//测试可以设置锁
        lock.l_type = F_WRLCK;
        lock_set (fd , &lock );
    }
        
    //释放锁
    lock.l_type = F_UNLCK ;
    lock_set (fd ,&lock );
    close (fd );
    return 0;
}

时间: 2024-10-06 02:39:38

linux c 笔记 文件(二)的相关文章

马哥Linux学习笔记之二——网络

1.MAC(Media Access Control,介质访问控制) 是解决底层数据通信冲突的解决方案.因此,给每一个接入这样网络的主机一个id标示符,这个id标示符就叫做MAC地址. 2.CSMA/CD(Carrier Sense Multipath Access Collision Detection,载波侦听多路访问冲突检测),线型网络 以太网Ethernet最核心的标志. 3.Token-Ring(IBM)技术 环形网络解决冲突的方案.但是由于IBM的专利限制没能推广开来. 4.星形网络

Linux学习笔记(二)——文件/目录/VIM

文件和目录管理 及 VI编辑器的使用 文件和目录管理,刚开始学这块的时候感觉内容很多很杂,但是学完进行总结后,发现其实很有条理的而且没什么难度,只是熟练掌握这些常用的命令就行了.至于Vim编辑器,不得不说,用了这个编辑器之后,感觉windows的notepad很没有技术含量了. 先简单总结一下文件和目录常用到的命令,简单的用法就略过.      文件操作命令:touch.file.which.find.cp.rm.mv.ln      文件内容操作命令:cat.more.less,head.ta

Linux学习笔记之二————Linux系统的文件和目录

一.Windows和Linux文件系统区别 1.在 windows 平台下,打开"计算机",我们看到的是一个个的驱动器盘符: 每个驱动器都有自己的根目录结构,这样形成了多个树并列的情形,如图所示:  2.在 Linux 下,我们是看不到这些驱动器盘符,我们看到的是文件夹(目录): 二.类Unix系统目录结构 1.ubuntu没有盘符这个概念,只有一个根目录/,所有文件都在它下面: 2.Linux 目录 /:根目录,一般根目录下只存放目录,在Linux下有且只有一个根目录.所有的东西都是

Linux学习笔记(二十)文件压缩 zip压缩、tar打包、打包、解包

一.zip压缩 首先安装zip与unzipyum install -y zip/unzip zip 1.txt.zip 1.txt 压缩文件1.txt,压缩文件名称为1.txt.zip zip -r 123.zip 123/ 压缩文件夹123/ 指定名称123.zip unzip 1.txt.zip zip压缩文件并不会删除源文件,解压时会提示是否覆盖已存在的文件 unzip 123.zip -d /root/456/ 解压123.zip文件到/root/456/ 目录下 unzip -l 12

Linux学习笔记(二)文件和目录管理

1.绝对路径"一定由根目录/写起":相对路径"不是由/写起". 2.特殊目录有.,..,-,~,~account,需要注意. 3.与目录相关的命令有cd,mkdir,rmdir,pwd等重要命令. 4.rmdir仅能删除空目录,要删除非空目录需使用"rm -r"命令. 5.用户能使用的命令是依据PATH变量所规定的的目录去查找的. 6.不同的身份(root与一般用户)系统默认的PATH并不相同.差异较大的地方在于/sbin和/usr/sbin.

Linux学习笔记(二)——Linux用户管理和权限管理

Linux系统的权限管理 Linux系统是一个多用户多任务的操作系统,多用户是指系统资源可以被不同用户各自拥有,即每个用户对自己的资源有特定的权限,用户之间互不影响.Linux系统有一套权限管理机制,文件不允许非授权用户访问或修改.这种机制的实现是通过用户和组的形式实现的. 5.1           Linux安全机制 账户管理是Linux安全机制的核心部分.登录Linux系统的用户都会被分配一个的用户账户.用户对系统上文件的访问权限取决于他们登录系统时使用的账户.每个用户的权限是通过创建用户

Linux学习笔记 (二)linux命令

一.命令行语法: 命令字 [选项][参数] 注意:Linux中对命令是区分大小写的. 二.获取命令帮助: 1.help命令:help xxx,shell内部指令,用来获取linux内部命令.例如:help help. 2.'--help'选项:xxx --help 获取命令的帮助信息. ls  --help. 3.man命令:man xxx 以全屏方式显示帮助手册,按q退出. man ls . 4.info命令:info xxx 以更详细的方式获取帮助手册 按q退出. info ls. 三.文件

linux c 笔记 文件(一)

一.linux下包含的文件类型 1.普通文件   2.目录文件   3.字符特殊文件    4.块特殊文件    5.FIFO    6.套接字   7.符号连接 二.linux的访问文件控制     chmod使用语法    $ chmod [options] mode[,mode] file1 [file2 ...]    使用ls命令的查看文件或目录的属性    $ ls -l file    chmod命令可以使用八进制数来指定权限.文件或目录的权限位是由9个权限位来控制,每三位为一组,

linux学习笔记(二)

Linux目录 Linux的文件系统采用级层式的树状结构,最上层的根目录是"/",根目录主要有以下目录. root目录:存放root用户的相关文件 home目录:系统默认的用户宿主目录,存放普通用户的相关文件,新增用户账号时,用户的宿主目录都存放在此目录下. bin目录:存放普通(常用)命令的目录 sbin目录:具有一定权限才可以使用的命令 porc目录: 此目录的数据都在内存中,如系统核心,外部设备,网络状态,由于数据都存放于内存中,所以不占用磁盘空间. mnt目录:默认挂载软驱光驱