perl使用find函数踩坑
2023-12-26 17:36:52
前言
写了一个脚本可以同时检查多个仿真log文件,并生成html表格。按照文件修改时间从新到旧排序。然后一直无法使用stath函数获取ctime。
结论:find函数会改变程序执行的当前目录,find(\&process_files, $dir);
函数是在$dir目录下运行的。
正文
测试环境的目录结构如下:
.
├── check_logs.pl
└── logs
├── 1.txt
├── 2.txt
└── 3.txt
1 directory, 4 files
一、使用内置函数stat
perl提供一个内置函数stat()获取文件相关信息,函数返回一个数组。
官方文档介绍stat - Perldoc 浏览器
my($dev, $ino, $mode, $nlink, $uid, $gid, $rdev, $size, $atime, $mtime, $ctime, $blksize, $blocks) = stat($file_name);
介绍几个比较重要的含义:
- $dev和$ino
- 文件所在设备的编号和文件的inode编号。
- $mode
- 文件的权限位集合,还包含其他信息位。低9位是linux的权限位。
- $nlink
- 文件或目录的硬连接数。
- $uid和$gid
- 以数值形式表示文件拥有者的用户ID和组ID
- $size
- 以字节为单位的文件大小
- $atime,$mtime和$ctime
- 三种时间戳,一个32位的整数,表示从1970年开始的秒数。
- 访问时间(atime):访问时间记录了文件最后一次被读取的时间。每当文件被读取时,其访问时间戳就会被更新。这对于某些应用程序来说是有用的,例如日志审计或跟踪文件的访问频率。
- 修改时间(mtime):修改时间记录了文件内容最后一次被修改的时间。当文件的内容(数据)被修改时,其修改时间戳就会被更新。这对于确定文件的最后修改时间非常有用。
- 更改时间(ctime):更改时间记录了文件元数据最后一次被更改的时间。元数据是与文件相关的非数据信息,例如文件的权限、所有者或文件类型等。当这些元数据属性发生变化时,其更改时间戳就会被更新。
先看下使用内置函数获取修改时间的代码
#! /bin/perl -w
use strict;
use warnings;
use File::Find;
use File::Basename;
my $time = (stat("./logs/1.txt"))[10];
print "$time\n";
运行结果如下:
[fengbh@VM-16-14-centos perl_stat]$ ./check_logs.pl
1703579691
二、使用File::stat
File::stat会覆盖内置的系统函数,它以类的方式提供类似内置函数stat的功能。
官方文档File::stat - by-name interface to Perl’s built-in stat() functions - Perldoc Browser
使用类的方式获取修改时间的代码如下:
#! /bin/perl -w
use strict;
use warnings;
use File::Find;
use File::stat;
my $time = stat("./logs/1.txt")->mtime;
print "$time\n";
运行结果如下:
[fengbh@VM-16-14-centos perl_stat]$ ./check_logs.pl
1703579691
三、在File::Find中使用
这里使用内置函数的方式实现。
#! /bin/perl -w
use strict;
use warnings;
use File::Find;
my $dir = "./logs";
find(\&process_files, $dir);
sub process_files{
return if !-f $_;
#debug
print "\$_ = $_\n";
print "\$File::Find::name = $File::Find::name\n\n";
#get mtime
my $mtime = (stat($File::Find::name))[10];
die "Can't stat file;$!\n" if !defined($mtime);
# debug
print "mtime = $mtime\n";
}
运行结果:
[fengbh@VM-16-14-centos perl_stat]$ ./check_logs.pl
$_ = 3.txt
$File::Find::name = ./logs/3.txt
Can't stat file;No such file or directory
运行发现报错,找不到文件。但是传给stat函数的文件路径名是正确的。
这是因为find函数会改变程序执行的当前目录,或者可以理解为process_files函数是在$dir目录下运行的。
这就是报错的原因,$File::Find::name是相对于初始执行目录的路径,$_才是相对于$dir的路径。
将代码修改为:
#! /bin/perl -w
use strict;
use warnings;
use File::Find;
my $dir = "./logs";
find(\&process_files, $dir);
sub process_files{
return if !-f $_;
#debug
print "\$_ = $_\n";
print "\$File::Find::name = $File::Find::name\n\n";
#get mtime
my $mtime = (stat($_))[10];
die "Can't stat file;$!\n" if !defined($mtime);
# debug
print "mtime = $mtime\n";
}
执行结果如下:
[fengbh@VM-16-14-centos perl_stat]$ ./check_logs.pl
$_ = 3.txt
$File::Find::name = ./logs/3.txt
mtime = 1703577429
$_ = 1.txt
$File::Find::name = ./logs/1.txt
mtime = 1703579691
$_ = 2.txt
$File::Find::name = ./logs/2.txt
mtime = 1703577426
执行结果正确
参考文献
文章来源:https://blog.csdn.net/weixin_44681954/article/details/135226741
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。 如若内容造成侵权/违法违规/事实不符,请联系我的编程经验分享网邮箱:veading@qq.com进行投诉反馈,一经查实,立即删除!
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。 如若内容造成侵权/违法违规/事实不符,请联系我的编程经验分享网邮箱:veading@qq.com进行投诉反馈,一经查实,立即删除!