APUE第四章学习笔记

前端之家收集整理的这篇文章主要介绍了APUE第四章学习笔记前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
/****************************************************
包含头文件:  #include <sys/stat.h>
结构体   struct stat
{
 mode_t st_mode; // 文件模式及类型
 ino_t st_ino; //inode节点号
 dev_t st_dev; //设备号
 dev_t st_rdev; //特殊设备号
 nlink_t st_nlink; //文件的连接数
 uid_t st_uid; //文件所有者用户ID
 gid_t st_gid; //文件所有者用户组ID
 off_t st_size; //普通文件对应的文件字节数
 struct timespec st_atime;// 文件最后被访问的时间
 struct timespec st_mtime;//文件内容最后被修改的时间
 struct timespec st_ctime;//文件状态改变时间
 blksize_t st_blksize; //文件内容对应的块大小
 blkcnt_t st_blocks; //文件被分配的块数量
}
*****************************************************/
/********************************************************
文件类型信息包含在stat结构体成员st_mode中
用下列宏(将st_mode作为参数)可确定文件类型:
文件类型
 S_ISREG()                              普通文件
 S_ISDIR()                              目录文件
 S_ISCHR()                              字符特殊文件
 S_ISBLK()                              块特殊文件
 S_ISFIFO()                             管道或FIFO
 S_ISLNK()                              符号链接
 S_ISSOCK()                             套接******************************************************/
/***************************************************
包含头文件:  #include <sys/stat.h>
函数原型:int stat(const char* restrict pathname,struct stat* restrict buf);
函数说明: 返回pathname命名的文件的有关的信息结构
返回值: 成功返回0出错返回-1
**************************************************/
/*********************************************************
包含头文件:  #include <sys/stat.h>
函数原型:   int fstat(int fd,struct stat* restrict buf);
函数说明: 获得已在描述符上打开文件的相关信息
返回值:
***************************************************/
/*************************************************
包含头文件:  #include <sys/stat.h>
函数原型:   int lstat(const char* restrict pathname,struct stat* buf);
函数说明: 类似于stat,但当命名的文件是一个符号链接时,lstat返回该符号链接的有关信息,而不是由该符号链接引用的文件信息
**************************************************/
/****************************************************
包含头文件:  #include <sys/stat.h>
函数原型:   int fstatat(int fd,const char* restrict pathname,struct stat* restrict buf,int flag);
函数说明:flag参数为AT_SYSLINK_nofollow时,fstatat不会跟随符号链接,而是直接返回符号链接的信息
*************************************************/

实例代码4.1.c

1 #include <sys/stat.h>
  2 #include <stdio.h>
  3 #include <stdlib.h>
  4 
  5 int main(int argc,char *argv[])
  6 {
  7     int i;
  8     struct stat buf;
  9     char *ptr;
 10 
 11     for (i = 1; i < argc; ++i)
 12     {
 13         printf("%s: ",argv[i]);
 14 
 15         if (lstat(argv[i],&buf) < 0)
 16         {
 17             printf("lstat error");
 18             continue;
 19         }
 20 
 21         if (S_ISREG(buf.st_mode))
 22             ptr = "普通文件";
 23 
 24         else if (S_ISDIR(buf.st_mode))
 25             ptr = "目录文件";
 26 
 27         else if (S_ISCHR(buf.st_mode))
 28             ptr = "字符特殊文件";
 29 
 30         else if (S_ISBLK(buf.st_mode))
 31             ptr = "块特殊文件";
 32 
 33         else if (S_ISFIFO(buf.st_mode))
 34             ptr = "管道或FIFO";
 35 
 36         else if (S_ISLNK(buf.st_mode))
 37             ptr = "符号链接";
 38 
 39         else if (S_ISSOCK(buf.st_mode))
 40             ptr = "套接字";
 41         else
 42             ptr = "unknown type";
 43         printf("类型: %s\n",ptr);
 44     }
 45     exit(0);
 46 }

运行:

/*****************************************
包含头文件:  #include <unsitd.h>
函数原型:   int access(const char* pathname,int mode);
函数说明:   按照实际用户ID和实际组ID进行访问权限测试
返回值: 若成功,返回0,若出错,返回-1 ---------------------------------------------------------------
包含头文件:  #include <unistd.h>
函数原型:int faccessat(int fd,const char* name,int mode,int flag);
函数说明:按实际用户ID和实际组ID进行访问权限测试
返回值:    若成功,返回0,若失败,返回-1;
参数说明: mode :若测试文件是否存在 mode就为F_OK
否则为下列值的或结果:   mode                 说明
 R_OK 测试读权限
 W_OK 测试写权限
 X_OK 测试执行权限
flag参数可以用于改变faccessat的行为,若flag设置为AT_EACCESS,访问检查用的是有效ID和有效组ID,而不是实际ID和实际组ID
注:若在这两种情况下:
 1.pathname是绝对路径
 2.fd为AT_FDCWD,pathname为相对路名
access 和faccessat效果是相同的,否则
faccessat计算打开目录(fd指向的)的相对路径pathname
**********************************************************/

实例4.2.c

1 #include <unistd.h>
  2 #include <fcntl.h>
  3 #include "apue.h"
  4 int main(int argc,char *argv[])
  5 {
  6     if (argc != 2)
  7      err_quit("usage: a.out <pathname>");
  8     if (access(argv[1],R_OK) < 0)
  9         err_ret("access error for %s",argv[1]);
 10     else
 11         printf("read access OK\n");
 12 
 13     if (open(argv[1],O_RDONLY) < 0)
 14         err_ret("open error for %s",argv[1]);
 15     else
 16         printf("open for reading ok\n");
 17 
 18     exit(0);
 19 }

实例:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>

int main(int argc,char* argv[])
{
    if (argc != 2)
    {
    printf("执行请加参数: <pathname>\n");
    exit(0);
    }

    if (access(argv[1],F_OK) < 0)
    {
    printf("%s 文件不存在\n",argv[1]);
    }
    else
    {
     if (access(argv[1],R_OK) < 0)
        printf("实际用户ID和实际组ID无读权限\n");
    else
        printf("实际用户ID和实际组ID有读权限\n");

    if (access(argv[1],W_OK) < 0)
        printf("实际用户ID和实际组ID无写权限\n");
    else
        printf("实际用户ID和实际组ID有写权限\n");

    if (access(argv[1],X_OK) < 0)
        printf("实际用户和实际组ID无执行权限\n");
    else
        printf("实际用户和实际组ID有执行权限\n");
    }
    if (faccessat(AT_FDCWD,argv[1],F_OK,AT_EACCESS) < 0)
    {
    printf("%s 文件不存在\n",argv[1]);
    }
    else
    {
    if (faccessat(AT_FDCWD,R_OK,AT_EACCESS) < 0)
        printf("有效用户ID和有效组ID无读权限\n");
    else
        printf("有效用户ID和有效组ID有读权限\n");

    if (faccessat(AT_FDCWD,W_OK,AT_EACCESS) < 0)
        printf("有效用户ID和有效组ID无写权限\n");
    else
        printf("有效用户ID和有效组ID有写权限\n");

    if (faccessat(AT_FDCWD,X_OK,AT_EACCESS) < 0)
        printf("有效用户ID和有效组ID无执行权限\n");
    else
        printf("有效用户ID和有效组ID有执行权限\n");
    }
    exit(0);
}

运行:

/************************************************************
包含头文件:  #include <sys/stat.h>
函数原型:   int chmod(const char* pathname,mode_t mode);
函数说明:   更改现有文件的访问权限
返回值:    若成功返回0,若出错返回-1
注:  为了改变一个文件的权限位,进程的有效ID必须等于文件所有者ID,或者该进程必须具有超级用户权限
************************************************************/
/****************************************************
包含头文件:  #include <sys/stat.h>
函数原型:   int fchmod(const char* pathname,mode_t mode);
函数说明:   对已打开的文件进行操作
返回值: 若成功,若失败返回-1
*****************************************************/
/***********************************************************
二、设置用户ID位和设置组ID位。
在文件模式字(st_mode)中,有两个位(bit10和bit11)分别称为设置用户组ID位和设置用户ID位,分别有两个测试常量S_ISGID和S_ISUID与其对应。若此文件为可执行文件:
(1)当设置用户组ID位为1,则文件执行时,内核将其进程的有效用户组ID设置为文件的所有组ID。
(2)当设置用户ID位为1,则文件执行时,内核将其进程的有效用户ID设置为文件所有者的用户ID。
*****************************************************/
/***************************************************
包含头文件:  #include <sys/stat.h>
函数原型:  int  fchmodat(int fd,const char* pathname,mode_t mode,int flag);
函数说明:   在下面两种情况:
 1.pathname参数为绝对路径
 2.fd取值为AT_FDCWD,pathname为相对路径
fchmodat和chmod和效果相同
否则,fchmodat相对于打开目录(由fd指向)pathname
参数flag可该fchmodat函数的行为,当设置了AT_SYMLINK_nofollow标志时,fchmodat并不会
跟随符号链接
**********************************************/
/*****************************************
包含头文件:  #include <unistd.h>
函数原型:int chown(const char* pathname,uid_t owner,gid_t group);
函数说明: 用于更改文件用户ID和组ID,如果两个参数owner或group任意一个是-1,则对应ID不变
返回值: 若成功,返回0,若失败-1
*****************************************/

/****************************************
包含头文件:  #include <unistd.h>
函数原型:   int fchown(int fd,gid_t group);
函数说明:   fchown用于由fd指向的已打开文件,改变其所有者(既然它作用于已打开文件,那么此函数就不能用于改变符号链接的所有者)
*****************************************************/
/****************************************************
包含头文件:  #include<unistd.h>
函数原型: int fchownat(int fd,gid_t group);
函数说明:fchownat在下列两种情况:
 1.pathname为绝对路径
 2.fd取值为AT_FDCWD,pathname为相对路径
 和chown,lchown功能相同
如果参数flag设置了AT_SYSLINK_nofollow,则lchown和fchownat行为相同
如果清除了AT_SYSLINK_nofollow标志,则fchownat和chown行为相同
否则,fd指向打开目录,pathname为相对路径名,fchownat函数计算相对于打开目录的pathname
*******************************************************/

/*****************************************************
包含头文件:  #include <unistd.h>
函数原型:   int lchown(const char* pathname,gid_t group);
函数说明:   更改文件的所有者,在文件是符号链接的情况下,会修改符号链接的所有者,而不是修改符号链接所引用文件的所有者
******************************************************/



/****************************************************
注: 如果这些函数由非超级用户调用,则在成功返回时,
该文件的设置用户ID位和设置组ID位会被清除
***************************************************/

测试代码(注意设置用户ID位和设置ID位被清除)

4.6.c

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

int main(int argc,char* argv[])
{
    struct stat statbuf;

    if (argc != 2)
    {
    printf("add <pathname>");
    exit(0);
    }

    if (stat(argv[1],&statbuf) < 0)
    {
    printf("get stat error\n");
    exit(0);
    }

    if (chmod(argv[1],statbuf.st_mode | S_ISUID | S_ISGID) < 0)
    {
    printf("chmod error\n");
    exit(0);
    }

    if (lchown(argv[1],27,1000) < 0)
    {
    printf("lchown error\n");
    exit(0);
    }
    exit(0);
}

4.7.c

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

int main(int argc,char* argv[])
{

    if (argc != 2)
    {
    printf("add <pathname>");
    }
    struct stat statbuf;

    if (stat(argv[1],&statbuf) < 0)
    {
    printf("stat error\n");
    exit(0);
    }

    if (statbuf.st_mode & S_ISUID)
    {
    printf("文件有设置用户ID权限\n");
    }
    else
    printf("文件无设置用户ID权限\n");

    if (statbuf.st_mode & S_ISGID)
    printf("文件有设置用户组ID权限\n");
    else
    printf("文件无设置用户组ID权限\n");

    printf("用户ID:%d\n",statbuf.st_uid);
    printf("用户组ID: %d\n",statbuf.st_gid);
    exit(0);
}

运行:

/*****************************************
文件长度:   stat结构成员st_size表示以字节为单位的文件长度,此字段只对普通文件,目录文件和符号链接有意义。
注: 对于符号链接,文件长度是在文件名的实际字节数
(因为符号链接文件长度总是由st_size指示,所以它并不包含通常C语言用作名字结尾的null字节)
**********************************************************/
文件截断
/*********************************************************
包含头文件:  #include <unistd.h>
函数原型:   int truncate(const char* pathname,off_t length);
 int ftruncate(int fd,off_t length);
函数说明:   将一个现有文件长度截断为length,如果该文件以前长度大于length,则length以后的字节都不可访问,如果小于length,以前文件尾和新文件尾之间之间的数据读做0(也就是在文件里创建了个空洞)
******************************************************/

实例代码:

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

int main(int argc,char* argv[])
{
    if (argc != 3)
    {
    printf("add <pathname> <length>\n");
    exit(0);
    }

    off_t length = atoi(argv[2]);

    if (truncate(argv[1],length) < 0)
    {
    printf("truncate error\n");
    exit(0);
    }

    exit(0);
}

运行:

/********************************************************
包含头文件:  #include <unistd.h>
函数原型:   int link(const char* existingpath,const char* newpath);
 int linkat(int efd,const char* existingpath,int nfd,const char* newpath,int flag);
函数说明:   创建一个新目录项newpath,它引用现有文件existingpath.如果newpath已存在,则返回出错.只
创建newpath中的最后一个分量,路径中的其他部分应当已经存在
注:  当现有文件是符号链接时,可通过flag参数指定来控制linkat是创建指向符号链接链接,还是创建指向符号链接所指向文件链接
返回值:    若成功,返回0,若出错,返回-1
*******************************************************/
/********************************************************
包含头文件:#include <unistd.h>
函数原型:   int unlink(const char* pathname);
 int unlinkat(int fd,int flag);
函数说明:   删除目录项,并将由pathname所引用文件链接计数减一
注:为了解除对文件链接,必须对包含该目录项的目录具有写和执行权限。
如果对该目录设置了粘帖位,则对该目录必须具有写权限,并且具备下面三个条件之一:
 1.拥有该文件
 2.拥有该目录
 3.具有超级用户权限
当flag设置为AT_REMOVEDIR,unlinkat函数可以类似于rmdir一样删除目录,如果这个标志被清除,unlinkat与unlink执行相同操作
********************************************************/

实例:
4.9.1.c

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

int main(int argc,char* argv[])
{
    if (argc != 3)
    {
    printf("add <pathname> <pathname>\n");
    exit(0);
    }

    if (link(argv[1],argv[2]) < 0)
    {
    printf("link error");
    exit(0);
    }
    exit(0);
}

4.9.2.c

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

int main(int argc,char* argv[])
{
    if (argc != 2)
    {
    printf("add <pathname>\n");
    exit(0);
    }

    if (unlink(argv[1]) < 0)
    {
    printf("unlink error");
    exit(0);
    }

    exit(0);

运行:

/********************************************************
包含头文件:  #include <stdio.h>
函数原型:   int rename(const char* oldname,const char* newname);
 int renameat(int oldfd,const char* oldname,int newfd,const char* newname);
函数说明:(1)如果oldname指的是一个文件而不是目录,
那么为该文件或符号链接重命名。在这种情况下,如果
newname已存在,则它不能引用一个目录,此时,则现将该目录项删除,然后将oldname命名为newname
(对包含oldname和newname的目录,调用进程必须具有写权限)
(2)如果oldname指的是一个目录,那么为该目录重命名
,如果newname已存在,则newname必须引用一个目录,并且该目录应当是空目录(空目录指的是该目录只有.和..项,如果newname已存在(而且引用一个空目录),则现将其删除,然后将oldname重命名为newname.
注: 当为一个目录重命名时,newname不能包含oldname作为其路径前缀
(3)如若oldname或newname引用符号链接,则处理符号链接本身,而不是符号链接所引用的文件
******************************************************/

实例:

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

int main(int argc,char* argv[])
{
    if (argc != 3)
    {
    printf("add <oldname> <newname>\n");
    exit(0);
    }

    if (rename(argv[1],argv[2]) < 0)
    {
    printf("rename error\n");
    exit(0);
    }

    exit(0);
}

若newdir里有文件,那么olddir会被移动到上层目录,并重命名,无上层目录,则出错且此上层目录不能为/

/*****************************************
包含头文件:  #include <unistd.h>
函数原型:   int symlink(const char* actualpath,const
char* sympath);
 int symlinkat(const char* actualpath,int fd,const char*
sympath);
函数说明:   创建一个符号链接
注:  在创建此符号链接时,并不要求actualpath已经存在,并且,并不要求actualpath和sympath并不需要位于同一文件系统中。
返回值:若成功,返回0,若失败,返回-1
*****************************************/

/***********************************************************
包含头文件:  #include <unistd.h>
函数说明: ssize_t readlink(const char* restrict pathname,const char* restrict buf,size_t bufsize);
 ssize_t readlinkat(int fd,const char* restrict pathname,char *restrict buf,size_t bufsize);
函数说明:   打开该链接本身,并读该链接中的名字
(如果函数成功执行,则返回读入buf的字节数,在buf
中返回的符号链接内容不以null字节终止)
返回值: 若成功,返回读取的字节数,若失败返回-1
**********************************************************/

实例:
4.11.c

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

int main(int argc,char* argv[])
{
    if (argc != 3)
    {
    printf("add <actualpath> <sympath>\n");
    exit(0);
    }

    if (symlink(argv[1],argv[2]) < 0)
    {
    printf("symlink error\n");
    exit(0);
    }

    exit(0);
}

4.12.c

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#define BUFFERSIZE 4096

int main(int argc,char* argv[])
{
    if (argc != 2)
    {
    printf("add <sympath>\n");
    exit(0);
    }

    char buf[BUFFERSIZE];
    int n;

    if ((n = readlink(argv[1],buf,BUFFERSIZE)) < 0)
    {
    printf("readlink error\n");
    exit(0);
    }

    printf("sympath actual content: %s\n",buf);
    exit(0);
}
/******************************************************
包含头文件:  #include <sys/stat.h>
函数原型:   int futimens(int fd,const struct timespec[2]);
int utimensat(int fd,const char* path,const struct timespec times[2],int flag);
函数说明:   更改文件的访问和修改时间
注:  (1)如果times参数是空指针,则访问时间和修改时间
都设置为当前时间
(2)如果times是非空指针,并且任一tv_nsec字段的值既不是UTIME_NOW也不是UTIME_OMIT,则进程的有效用户ID必须等于该文件所有者的ID,或者进程必须是一个超级用户进程,对文件只具有写权限是不够的
(3) futimens函数需要打开文件来更改它的时间,utimensat的flag参数如果设置了AT_SYMLINK_nofollow,则符号链接本身会被修改,否则默认下跟随符号链接,并把文件的时间改成符号链接的时间

*********************************************************/
/**********************************************************
包含头文件:  #include <sys/stat.h>
函数原型:   int mkdir(const char* pathname,mode_t
mode);
int mkdirat(int fd,mode_t mode);
函数说明:   创建一个空目录,其中.和..目录项是自动创建的,所指定的文件访问权限mode由进程的文件模式创建屏蔽修改
注:对于目录通常至少要设置一个执行权限位,以访问该目录中的文件名
返回值:若成功,若失败,返回-1
***********************************************************/

/**********************************************************
包含头文件:  #include <unistd.h>
函数原型:   int rmdir(const char* pathname);
函数说明:   删除空目录,空目录是指只包含.和..这两项的目录
返回值:    若成功,若出错,返回-1
**********************************************************/

实例:

#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>

#define MODE (S_IRWXU | S_IRWXG | S_IRWXO)
int main(int argc,char* argv[])
{
    if (argc != 2)
    {
    printf("add <dirname>\n");
    exit(0);
    }

    umask(0);

    if (mkdir(argv[1],MODE) < 0)
    {
    printf("mkdir error\n");
    exit(0);
    }

    exit(0);
}

运行:

/********************************************************
包含头文件:  #include <dirent.h>
函数原型:   DIR *opendir(const char pathname);
 DIR *fdopendir(int fd);
函数说明:   打开目录
返回值:    若成功,返回DIR指针,返回NULL
********************************************************/

/********************************************************
包含头文件:  #include <dirent.h>
函数原型:   struct dirent* readdir(DIR *dp);
函数说明: 读取目录
返回值:    返回参数dp目录流的下个目录项进入点(指针),到达目录项结尾返回NULL
*******************************************************/

/******************************************************
包含头文件:  #include <dirent.h>
函数原型:   void rewinddir(DIR *dp);
函数说明: 设置参数dp目前读取的位置为原来开头的读取位置
*******************************************************/




/*********************************************************
包含头文件:  #include <dirent.h>
函数原型:   int closedir(DIR* dp);
函数说明: 关闭目录
返回值:    若成功,返回-1
*********************************************************/

/*******************************************************
包含头文件:  #include <dirent.h>
函数原型:   long telldir(DIR *dp);
函数说明:与dp关联目录的当前位置
******************************************************/

/**********************************************************
包含头文件:  #include <dirent.h>
函数原型:   void seekdir(DIR *dp,long loc);
函数说明:   设置读取目录参数dp当前位置,loc为偏移量,则下次readdir从新当前位置开始
***********************************************************/

实例: 4.15.cpp

#include <iostream>
#include <dirent.h>
#include <stdlib.h>
#include <stdexcept>
using namespace std;

int main(int argc,char* argv[])
{
    try
    {
    if (argc != 2)
        throw domain_error("add <dirname>\n");
    DIR *dp;

    if ((dp = opendir(argv[1])) == NULL)
        throw domain_error("dirname is no sense\n");

    struct dirent* dt;
    long offset;

    while ((dt = readdir(dp)) != NULL)
    {
        offset = telldir(dp);
        cout <<"目录第" << offset << "项" << dt->d_name << endl;
    }
    }
    catch(const exception& e)
    {
    cout << e.what() << endl;
    }

}
/***********************************************************
包含头文件:  #include <unistd.h>
函数原型:   int chdir(const char* pathname);
 int fchdir(int fd);
函数说明:   分别用pathname或打开文件描述符来指定新的当前工作目录
返回值:    若成功,返回0,若失败,返回-1
***********************************************************/

/***********************************************************
包含头文件:  #include <unistd.h>
函数原型:   char* getcwd(char* buf,size_t size);
函数说明:   得到当前工作目录路径名
返回值: 若成功,返回buf,若失败,返回NULL
***********************************************************/

实例:

4.17.c

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

int main()
{
    if (chdir("/home/marco/o") < 0)
    {
    printf("chdir error\n");
    exit(0);
    }

    printf("chdir to /home/marco/o\n");
    exit(0);
}

4.18.c

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

int main()
{
    char *ptr;
    size_t size;

    if (chdir("/home/marco/o") < 0)
    {
    printf("chdir error \n");
    exit(0);
    }

    ptr = (char*)malloc(50);
    size = 50;
    if (getcwd(ptr,size) == NULL)
    {
    printf("getcwd Failed\n");
    exit(0);
    }

    printf("cwd = %s\n",ptr);
    exit(0);
}

运行:

猜你在找的Bash相关文章