Linux学习——目录操作和库使用

目录

一、打开目录

二、读取目录

三、关闭目录

四、修改文件权限

五、获取文件属性

六、库的概念:

1、静态库

缺点:

优点:

创建静态库步骤:

链接静态库:

 2、动态库

动态库的生成步骤:

练习题:


一、打开目录

#include  <dirent.h>

 DIR  *opendir(const char *name);

 DIR *fdopendir(int fd);  使用文件描述符,要配合open函数使用

DIR是用来描述一个打开的目录文件的结构体类型

成功时返回目录流指针;出错时返回NULL

二、读取目录

#include  <dirent.h>

 struct  dirent *readdir(DIR *dirp);

struct dirent是用来描述目录流中一个目录项的结构体类型

包含成员char  d_name[256]   参考帮助文档

成功时返回目录流dirp中下一个目录项;

出错或到末尾时时返回NULL

三、关闭目录

closedir函数用来关闭一个目录文件:

 #include  <dirent.h>

 int closedir(DIR *dirp);

成功时返回0;出错时返回EOF

实验程序:

#include <dirent.h>
#include <stdio.h>
int main(int argc,char **argv){

    DIR* dp;
    struct dirent *dt;
    dp=opendir("/home/book/Desktop/Lv4/Day5");
    if(dp<0){
        perror("opendir");
        return 0;
    }
    
    while((dt=readdir(dp))!=NULL){
       printf("%s\n",dt->d_name);
    }
    

    closedir(dp);



}

实验结果:

 

四、修改文件权限

chmod/fchmod函数用来修改文件的访问权限:

 #include  <sys/stat.h>

 int  chmod(const char *path, mode_t mode);

 int  fchmod(int fd, mode_t mode);

成功时返回0;出错时返回EOF

注意:在vmwarewindows共享的文件夹下,有些权限不能改变。

五、获取文件属性

#include  <sys/stat.h>

 int  stat(const char *path, struct stat *buf);

 int  lstat(const char *path, struct stat *buf);

 int  fstat(int fd, struct stat *buf);

 

成功时返回0;出错时返回EOF

如果path是符号链接stat获取的是目标文件的属性;而lstat获取的是链接文件的属性

 

 

 

实验程序:

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <time.h>

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

   struct stat buf;
   int ret;
   ret = stat("chmod_t.c",&buf);
   if(ret<0){
      perror("stat");
      return 0;

   }
   if(S_ISREG(buf.st_mode)){
       printf("-");
   }
   if(S_ISDIR(buf.st_mode)){
       printf("d");
   }
   if(S_ISCHR(buf.st_mode)){
       printf("c");
   }
   if(S_ISBLK(buf.st_mode)){
       printf("b");
   }
   if(S_ISFIFO(buf.st_mode)){
       printf("p");
   }
   if(S_ISSOCK(buf.st_mode)){
       printf("s");
   }
   
//   printf(" ");
   int i;
   for(i=8;i>=0;i--){
       if(buf.st_mode & (1<<i)){
          switch(i%3){
          case 2:
              printf("r");
              break;
          case 1:
              printf("w");
              break;
          case 0:
              printf("x");
              break;
          }
       }else{
           printf("-");
       }


   }
   
   printf(" %d",(int)buf.st_size);

   struct tm *t;
   t = localtime(&buf.st_ctime);
   printf(" %d-%d-%d %d:%d",t->tm_year+1900,t->tm_mon+1,t->tm_mday,t->tm_hour,t->tm_min);

   printf(" chmod_t.c\n");


   

}

实验结果:

 

六、库的概念:

区别:

1、静态库的扩展名一般为“.a”或“.lib”;动态库的扩展名一般为“.so”或“.dll”。

2、静态库在编译时会直接整合到目标程序中,编译成功的可执行文件可独立运行;动态库在编译时不会放到连接的目标程序中,即可执行文件无法单独运行。

从产品化的角度,发布的算法库或功能库尽量使动态库,这样方便更新和升级,不必重新编译整个可执行文件,只需新版本动态库替换掉旧动态库即可。

从函数库集成的角度,若要将发布的所有子库(不止一个)集成为一个动态库向外提供接口,那么就需要将所有子库编译为静态库,这样所有子库就可以全部编译进目标动态库中,由最终的一个集成库向外提供功能。

1、静态库

缺点:

浪费磁盘空间,1000个文件调用就要拷贝1000份。

升级后需要重新编译链接。

优点:

编译(链接)时把静态库中相关代码复制到可执行文件中

程序中已经包含代码,运行时不再需要静态库,运行速度更快

创建静态库步骤:

1 . 编写库文件代码,编译为.o 目标文件。

2.ar 命令 创建  libxxxx.a 文件

   ar -rsv  libxxxx.a  xxxx.o

   注意:

1 静态库名字要以lib开头,后缀名为.a

2 没有main函数的.c 文件不能生成可执行文件。

链接静态库:

gcc -o 目标文件 源码.c  -L路径  -lxxxx

-L  表示库所在的路径

-l 后面跟库的名称

程序:

test.c

#include <stdio.h>

void hello();
void bye();
int main(int argc,char **argv){
    hello();
    //bye();
    return 0;
}

hello.c

#include <stdio.h>
void bye(){
   printf("bye\n");

}

结果:

 2、动态库

动态库的生成步骤:

1、生成位置无关代码的目标文件

 gcc  -c  -fPIC  xxx.c xxxx.c ....

2、生成动态库

 gcc  -shared -o libxxxx.so  xxx.o  xxx.o ....

3、编译可执行文件

gcc -o 目标文件 源码.c  -L路径  -lxxxx

执行动态库的可执行文件错误

./test: error while loading shared libraries: libmyheby.so: cannot open shared object file: No such file or directory

含义:可执行文件所使用的动态库找不到

解决办法:

因为动态库会先在系统的lib里找所以找到动态库,添加到/usr/lib里面

//不提倡因为会把自己的lib库搞乱

或者使用export  LD_LIBRARY_PATH=$LD_LIBRARY_PATH:你的动态库目录

添加在~/.bashrc 文件里面

使用source ~/.bashrc 生效。

实验程序:

bye.c

#include <stdio.h>
void bye(){
   printf("bye\n");

}

实验结果:

 换一个终端会再次失效

加入到系统环境变量里才能一直生效。

 

查看可执行文件使用的动态库:

ldd 命令 :   ldd 你的可执行文件

root@haas-virtual-machine:/mnt/hgfs/share/newIOP# ldd test

        linux-vdso.so.1 =>  (0x00007fff6548d000)

        libmyheby.so => /mnt/hgfs/share/newIOP/day5/libmyheby.so (0x00007f5c89521000)

        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f5c89144000)

        /lib64/ld-linux-x86-64.so.2 (0x000055fe52211000)

root@haas-virtual-machine:/mnt/hgfs/share/newIOP/day5# ldd test

        linux-vdso.so.1 =>  (0x00007ffcb652c000)

        libmyheby.so => not found

        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fbeeffaf000)

        /lib64/ld-linux-x86-64.so.2 (0x0000561003c3b000)

练习题:

遍历一个文件夹下所有文件,并打印文件大小和日期

程序:

#include <dirent.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <time.h>
#include <stdio.h>
#include <string.h>
int main(int argc,char **argv){

    DIR* dp;
    struct dirent *dt;
	struct stat buf;
	char buf1[64];
	char buf2[15];
	//memset(buf2,' ',15);
    int ret;
	struct tm *t;
    dp=opendir("/home/book/Desktop/Lv4/Day5");
    if(dp<0){
        perror("opendir");
        return 0;
    }
    
    while((dt=readdir(dp))!=NULL){
		//printf("%s ",dt->d_name);
		ret = stat(dt->d_name,&buf);
		strcpy(buf2,dt->d_name);
   		if(ret<0){
       		perror("stat");
       		return 0;
   		} 
		//printf(" %d",(int)buf.st_size);
   		t = localtime(&buf.st_ctime);
   		//printf(" %d-%d-%d %d:%d",t->tm_year+1900,t->tm_mon+1,t->tm_mday,t->tm_hour,t->tm_min);
		sprintf(buf1,"name:%15s size:%5d B time: %d-%d-%d %d:%d\n",buf2,(int)buf.st_size,t->tm_year+1900,t->tm_mon+1,t->tm_mday,t->tm_hour,t->tm_min);
		printf("%s",buf1);
    }
    
    closedir(dp);

}

结果:

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

宇努力学习

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值