unix系统编程小结(二)------文件和目录

简介:
一.对linux的安全机制的一点感悟
         各种权限,read,write,execute,set-user-ID,set-group-ID,sticky bit,对目录的权限,对文件的权限,用于保证系统安全的各种组合技,各种经典。比如,如果我们想unlink一个文件,就必须拥有该文件所在目录的write与execute的权限。
二.两个小例子
         1.当文件有hole时,cp命令会同时拷贝这些hole为'\0'。这里是一个实现了拷贝时跳过文件hole的程序。ps:我用的buffer是一个字节的,效率很低,但如果用大的buffer就会使得hole被移除,使得原先分开的字符被连上。我没想好如何解决这个问题。如果您知道,请您告诉小弟我,非常感谢!
[cpp]  view plain copy
  1. #include <apue.h>  
  2. #include <my_error.h>  
  3. #include <fcntl.h>  
  4.   
  5. int main()  
  6. {  
  7.     char buf[1];  
  8.     int fd,fd_to;  
  9.     int n;  
  10.     if( (fd=open("temp_in_hole",O_RDWR) )<0)  
  11.         err_sys("error open");  
  12.     if( (fd_to=open("temp_OUT_hole",O_WRONLY))<0 )  
  13.         err_sys("error open for ");  
  14.     while( (n=read(fd,buf,1))!=0 )  
  15.     {  
  16.         if(buf[0]!='\0')   
  17.             if(write(fd_to,buf,1)!=n)  
  18.                 err_sys("error write");  
  19.     }  
  20.     close(fd);  
  21.     close(fd_to);  
  22.     return 0;  
  23. }  

         2.遍历目录。这里只贴出主要代码:   ps:有一个技巧就是每遇到一个目录时,就用chdir将该目录设置为当前的工作目录,可以提高程序运行的效率。
[cpp]  view plain copy
  1. static int dopath(Myfunc * func)  
  2. {  
  3.     struct stat    statbuf;  
  4.     struct dirent  *dirp;  
  5.     DIR            *dp;  
  6.     int            ret;  
  7.     char           *ptr;  
  8.   
  9.     if(lstat(fullpath,&statbuf)<0)  
  10.         return (func(fullpath,&statbuf,FTW_NS));  
  11.     if(S_ISDIR(statbuf.st_mode)==0)  
  12.         return (func(fullpath,&statbuf,FTW_F));  
  13.   
  14.     if( (ret=func(fullpath,&statbuf,FTW_D))!=0 )  
  15.         return ret;  
  16.   
  17.     ptr=fullpath+strlen(fullpath);  
  18.     *ptr++='/';  
  19.     *ptr=0;  
  20.   
  21.     if(chdir(fullpath)<0)  
  22.         err_ret("chdir for %s failed!",fullpath);  
  23.     if((dp=opendir("./"))==NULL)  
  24.         return (func(fullpath,&statbuf,FTW_DNR));  
  25.     while((dirp=readdir(dp))!=NULL)  
  26.     {  
  27.         if(strcmp(dirp->d_name,".")==0 || strcmp(dirp->d_name,"..")==0)   
  28.             continue;  
  29.         strcpy(ptr,dirp->d_name);  
  30.         if(ret=dopath(func)!=0)  
  31.             return ret;  
  32.     }  
  33.     ptr[-1]=0;  
  34.     if(closedir(dp)<0)  
  35.         err_ret("can't close directory %s",fullpath);  
  36.     if(chdir("../")<0)  
  37.         err_ret("chdir for ../ failed!");  
  38.     return ret;  
  39. }  

三.一些笔记
         相关的system call很多,具体的用法我就不列出来了,只列出一些我在看书过程中的笔记(有些是摘自apue)。
         1.there is no distinction to the UNIX kernel whether this data is text or binary.
    2.FIFO is a type of file used for communication between processed.It's sometimes called a named pipe.
         3.if we used the stat function,we would never see symbolic links.用lstat才能处理symbolic links.
    4. set-user-ID:这个概念要特别注意。比如说,当一个文件被设置了set-user-ID后,而这个文件的owner是root,然后用一个非root的进程执行这个文件,那么该文件所执行的进程也将拥有root权限。比如说passwd命令就是一个set-user-ID程序。任何用户都可以修改密码,但/etc/passwd只能是root才有write的权利,所以这时set-usr_ID就可以发挥作用,让所有执行passwd的进程都可以向/etc/passwd中write。
         5.whenever we want to open any type of file by name ,we must have execute permission in each directory mentioned in the name,including the current directory,if it is implied.This is why the execute permission bit for a directory is often called the search bit.这个比较好理解,对于一个目录来说,自然不可能去执行目录,所以目录的可执行性就是为了执行该目录下的文件作基础。
    6.目录的read权限仅仅是获得a list of all the filenames in the directory.目录的read权限与执行权限是很不相同的。
    7.we cannot create a new file in a directory unless we have write permission and execute permission in the directory.
    8.To delete an existing file,we need write permission and execute permission in the directory containing the file.We do not need read permission or write permission for the file itself.以前没有注意目录与文件之间的这样的权限的联系。
    9.当用gcc编译生成一个二进制文件后,用chmod u+s来设置set-user-ID。 当再用gcc更新该二进制文件后,set-user-ID就会消失
    10.read系统调用读的内容是不会自动加上'\0'的,所以在使用printf输出该内容的时候要手动加上'\0'
    11.open(filename,O_TRunC).就可以清空文件filename
    12.The i-nodes are fixed-length entries that contain most of the information about a file.Most of the information in the stat structure is obtained from the i-node.
    13.对目录做硬链接有可能使得文件系统出现循环,而这个循环是处理文件系统的很多工具都无法解决的。所以非常多的UNIX实现版本都不允许对目录做硬链接。apue上4.16有出现循环的例子
    14.We've mentioned before that to unlink a file,we must have write permission and execute permission in the directory containing the directory entry,as it is the directory entry that we will be removing.Also,we mentioned in Section 4.10(in apue)that if the sticky bit is set in this directory we must have write permission for the directory and one of the following:Own the file;Own the directory;Have superuser privileges.
    15.A symbolic link is an indirect pointer to a file,unlike the hard links,which pointed directly to the i-node of the file.
    16.
?
ln -s ../foo link_file   这里的../foo可以是不存在的文件,然后就像绝对路径一样,link_file 直接指向../foo。
例子:foo是一个目录。
ln -s ../foo foo/testdir
再ll  foo会显示
drwxr-xr-x 2 root root 4096 2012-12-06 22:37 ./
drwxr-xr-x 3 root root 4096 2012-12-06 22:37 ../
lrwxrwxrwx 1 root root    6 2012-12-06 22:37 testdir -> ../foo/
    17.All the information in the i-node is stored separately from the actual contents of the file.
    18.A directory is simply a file containing directory entries:filenames and associated i-node numbers.
    19.
?
typedef 的一个用法:  
  typedef  int  Func( int  a, int  b);   //这声明了一个新的函数类型,Func,这种函数类型有两个int类型的参数,返回值也是int类型
  static  Func  my_func1;    //my_func1的函数原型,完整的写法是static int my_func1(int a ,int b);  在这里使用typedef,会使得代码更加简洁,清晰,对代码的可读性有好处
    20.
?
char  *p;   *p++= '0' ;  是将*p赋值为 '0' 后,p指针再加1个单位的地址
                  *++p= '0' ;  是先将p执行加一个单位的地址后,再将*p赋值为 '0' ;   这是一个比较容易混淆和容易记错的地方
    21.char *ptr;  ptr[-1]是指是的ptr指向位置的上一个元素。
    22.The current working directory is an attribute of a process .
    23. If we try,using either open or creat,to create a file that alredy exists,the file's access permission bits are not changed but the files were truncated(被清空).  
    24.The size of a directory should never be 0,since there should always be entries for dot and dot-dot.The size of a symbolic link is the number of characters in the pathname contained in the symbolic link,and this pathname must always contain at least one character.
    25.time ./pro1 计算整个pro1程序执行的时间
    26.The kernel has no inherent limit on the depth of a directory tree.But many commands will fail on pathnames that exceed PATH_MAX,比如getcwd.
   参考资料:apue
    如果您觉得我的文章对您有帮助,请您赞一下,非常感谢!

本文转自NeilHappy 51CTO博客,原文链接:http://blog.51cto.com/neilhappy/1083055,如需转载请自行联系原作者
相关文章
|
8月前
|
存储 Shell Linux
【Shell 命令集合 网络通讯 】Linux 显示Unix-to-Unix Copy (UUCP) 系统的状态信息 uustat命令 使用指南
【Shell 命令集合 网络通讯 】Linux 显示Unix-to-Unix Copy (UUCP) 系统的状态信息 uustat命令 使用指南
88 0
|
3月前
|
算法 Unix 数据安全/隐私保护
Python编程--UNIX口令破解机
Python编程--UNIX口令破解机
32 1
|
5月前
|
开发框架 Unix Linux
LangChain 构建问题之在Unix/Linux系统上设置OpenAI API密钥如何解决
LangChain 构建问题之在Unix/Linux系统上设置OpenAI API密钥如何解决
60 0
|
7月前
|
Unix
Unix环境高级编程(第三版)中apue.h头文件及其依赖安装教程
Unix环境高级编程(第三版)中apue.h头文件及其依赖安装教程
131 0
|
8月前
|
Oracle 关系型数据库 Unix
SAP系统拷贝 UNIX + Oracle
SAP系统拷贝 UNIX + Oracle
64 1
|
8月前
|
安全 Unix Linux
【专栏】`rmdir`命令在Linux和类Unix系统中用于删除空目录,不适用于非空目录
【4月更文挑战第28天】`rmdir`命令在Linux和类Unix系统中用于删除空目录,不适用于非空目录。基本语法为`rmdir [options] directory...`,常用选项包括`-p`(递归删除空父目录)和`--ignore-fail-on-non-empty`(忽略非空目录错误)。与`rm -r`相比,`rmdir`更安全,适用于知道目录为空的情况。在自动化脚本和清理构建目录等场景中,`rmdir`能有效管理空目录。使用时确保目录为空,避免误删,必要时结合`ls`和`sudo`检查或提升权限。
111 1
|
8月前
|
Unix Linux Shell
在Unix/Linux系统中,文件和目录的权限管理
在Unix/Linux系统中,文件和目录的权限管理
100 3
|
Unix Shell Python
unix高级编程-fork和execve
unix高级编程-fork和execve
65 0
|
Ubuntu Unix Shell
unix高级编程-fork之后父子进程共享文件
unix高级编程-fork之后父子进程共享文件
66 0
|
Unix Shell 程序员
《UNIX环境高级编程(第3版)》——1.4 文件和目录
目录(directory)是一个包含目录项的文件。在逻辑上,可以认为每个目录项都包含一个文件名,同时还包含说明该文件属性的信息。文件属性是指文件类型(是普通文件还是目录等)、文件大小、文件所有者、文件权限(其他用户能否访问该文件)以及文件最后的修改时间等。
2539 0