perl使用find函数踩坑

前言

写了一个脚本可以同时检查多个仿真log文件,并生成html表格。按照文件修改时间从新到旧排序。然后一直无法使用stath函数获取ctime。

结论:find函数会改变程序执行的当前目录find(\&process_files, $dir);函数是在$dir目录下运行的

正文

测试环境的目录结构如下:

txt 复制代码
.
├── check_logs.pl
└── logs
    ├── 1.txt
    ├── 2.txt
    └── 3.txt

1 directory, 4 files

一、使用内置函数stat

perl提供一个内置函数stat()获取文件相关信息,函数返回一个数组。

官方文档介绍stat - Perldoc 浏览器

perl 复制代码
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):更改时间记录了文件元数据最后一次被更改的时间。元数据是与文件相关的非数据信息,例如文件的权限、所有者或文件类型等。当这些元数据属性发生变化时,其更改时间戳就会被更新。

先看下使用内置函数获取修改时间的代码

perl 复制代码
#! /bin/perl -w
use strict;
use warnings;
use File::Find;
use File::Basename;

my $time = (stat("./logs/1.txt"))[10];
print "$time\n";

运行结果如下:

bash 复制代码
[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

使用类的方式获取修改时间的代码如下:

perl 复制代码
#! /bin/perl -w
use strict;
use warnings;
use File::Find;
use File::stat;

my $time = stat("./logs/1.txt")->mtime;
print "$time\n";

运行结果如下:

bash 复制代码
[fengbh@VM-16-14-centos perl_stat]$ ./check_logs.pl 
1703579691

三、在File::Find中使用

这里使用内置函数的方式实现。

perl 复制代码
#! /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";
}

运行结果:

bash 复制代码
[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的路径。

将代码修改为:

perl 复制代码
#! /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";
}

执行结果如下:

bash 复制代码
[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

执行结果正确

参考文献

  1. 官方文档介绍stat - Perldoc 浏览器
  2. 官方文档File::stat - by-name interface to Perl's built-in stat() functions - Perldoc Browser
  3. 《perl语言入门》
相关推荐
橘子真甜~24 分钟前
C/C++ Linux网络编程6 - poll解决客户端并发连接问题
服务器·c语言·开发语言·网络·c++·poll
9***Y481 小时前
Java开发工具IntelliJ IDEA技巧
java·开发语言·intellij-idea
码力码力我爱你1 小时前
C++性能基准测试
开发语言·c++
张人玉1 小时前
C#WPF——MVVM框架编写管理系统所遇到的问题
开发语言·c#·wpf·mvvm框架
java1234_小锋1 小时前
讲讲Mybatis的一级、二级缓存?
java·开发语言·mybatis
z***I3942 小时前
JavaScript原型链
开发语言·前端·javascript
x***58702 小时前
JavaScript语音识别开发
开发语言·javascript·语音识别
小年糕是糕手2 小时前
【C++】C++入门 -- 输入&输出、缺省参数
c语言·开发语言·数据结构·c++·算法·leetcode·排序算法
q***92512 小时前
PHP搭建开发环境(Windows系统)
开发语言·windows·php
Chrison_mu2 小时前
Android项目背景动效-Kotlin
android·开发语言·kotlin