当前位置: 首页 > news >正文

perl使用find函数踩坑

前言

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

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

正文

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

.
├── check_logs.pl
└── logs├── 1.txt├── 2.txt└── 3.txt1 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 $_;#debugprint "\$_ = $_\n";print "\$File::Find::name = $File::Find::name\n\n";#get mtimemy $mtime = (stat($File::Find::name))[10];die "Can't stat file;$!\n" if !defined($mtime);# debugprint "mtime = $mtime\n";
}

运行结果:

[fengbh@VM-16-14-centos perl_stat]$ ./check_logs.pl 
$_ = 3.txt
$File::Find::name = ./logs/3.txtCan'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 $_;#debugprint "\$_ = $_\n";print "\$File::Find::name = $File::Find::name\n\n";#get mtimemy $mtime = (stat($_))[10];die "Can't stat file;$!\n" if !defined($mtime);# debugprint "mtime = $mtime\n";
}

执行结果如下:

[fengbh@VM-16-14-centos perl_stat]$ ./check_logs.pl 
$_ = 3.txt
$File::Find::name = ./logs/3.txtmtime = 1703577429
$_ = 1.txt
$File::Find::name = ./logs/1.txtmtime = 1703579691
$_ = 2.txt
$File::Find::name = ./logs/2.txtmtime = 1703577426

执行结果正确

参考文献

  1. 官方文档介绍stat - Perldoc 浏览器
  2. 官方文档File::stat - by-name interface to Perl’s built-in stat() functions - Perldoc Browser
  3. 《perl语言入门》
http://www.lryc.cn/news/269481.html

相关文章:

  • Java IDEA JUnit 单元测试
  • 深入理解 c++ 函数模板
  • 系列十二、Linux中安装Zookeeper
  • k8s之陈述式资源管理
  • 7天玩转 Golang 标准库之 http/net
  • 钡铼技术集IO数据采集可编程逻辑控制PLC无线4G环保物联网关
  • STM32CubeMX教程10 RTC 实时时钟 - 周期唤醒、闹钟A/B事件和备份寄存器
  • HarmonyOS4.0系统性深入开发08服务卡片架构
  • 002文章解读与程序——中国电机工程学报EI\CSCD\北大核心《计及源荷不确定性的综合能源生产单元运行调度与容量配置两阶段随机优化》已提供下载资源
  • Typora快捷键设置详细教程
  • 《异常检测——从经典算法到深度学习》25 基于深度隔离林的异常检测算法
  • 第7章 1 异常处理
  • 昇腾910平台安装驱动、固件、CANN toolkit、pytorch
  • 【数据挖掘】模型融合
  • DM、Oracle、GaussDB、Kingbase8(人大金仓数据库)和HIVE给列增加注释
  • C语言实例_stdlib.h库函数功能及其用法详解
  • Error in onLoad hook: “URIError: URI malformed“ found in…报错处理以及完善uniapp针对对象传参
  • c语言-位操作符练习题
  • 园林机械部件自动化三维测量检测形位公差-CASAIM自动化三维检测工作站
  • o2o生活通全开源尊享版+多城市切换+企业付款+交友IM+平台快报
  • UE4开发BIM程序 的 流程
  • 【AI大语言模型】ChatGPT在地学、GIS、气象、农业、生态、环境等领域中的应用
  • 【面试题】写一个睡眠函数
  • 4. 云原生之kubesphere基础服务搭建
  • 思福迪运维安全管理系统 任意文件读取漏洞
  • OCR在审核应用落地
  • 借贷协议 Tonka Finance:铭文资产流动性的新破局者
  • Python+Yolov5+Qt交通标志特征识别窗体界面相片视频摄像头
  • 浅谈高并发以及三大利器:缓存、限流和降级
  • 深入ArkUI:深入实战组件text和text input