从函数复用开始:eval和do执行perl文件

当我们定义了一个功能比较通用的子程序,比如获取数值的绝对值。想要到处使用这个子程序,就得不断复制、粘贴这段绝对值函数的定义文本。显然,这是不太理想的方式。

于是,就将包含这个子程序的代码放进一个perl文件,然后通过特殊的语法去导入这个文件。

例如,文件sum.pm包含一个sum子程序,该子程序返回参数相加之和。

#!/usr/bin/env perl
# 注意,这里为了测试,没加use strict
sub sum {
my $sum;
$sum = map { $_ + $sum } $@
$name;
}
$name="longshuai"; # 全局变量属性

其中.pm后缀表示perl module。在perl 4的时候,使用的是.pl后缀,代表的是perl library。但现在,模块、包都使用.pm替代,而.pl主要代表perl程序。

eval语句导入文件

可以在其它perl文件中(如eval.pm)通过eval语句临时编译这个文件(sum.pm)中的语句。只不过在eval之前,需要先将sum.pm文件中的内容读取:

#!/usr/bin/env perl
#
use strict;
use warnings;
use 5.010; open my $fh,"<","sum.pm" or die "Can't open file: $!";
undef $/;
my $sum_code=<$fh>; # 读取代码保存到变量中
close $fh;
eval $sum_code; # eval评估编译这段代码,并执行
die $@ if $@; my $sum=sum(1,2,3); # 引用来自sum.pm的函数
say $sum;

上面的代码会报错。因为eval $sum_code是将来自sum.pm中的代码放在当前文件中临时进行编译,这段来自sum.pm的代码已经属于本文件。就等价于:

eval CODE;

所以,来自sum.pm中的全局属性$name在当前文件的use strict编译指示下将引发错误。所以,得将sum.pm内容中的$name去掉,或者加上my修饰。

eval中来自sum.pm的代码将能访问它所在代码块的词法变量。

do语句导入文件

也可以通过do语句临时编译这个文件,它将在当前程序(无论do语句是否是在代码块中)引入编译的结果(但除了子程序外的其它属性,由于一般会加上use strict,而导致为未声明的变量不可使用,间接地,所导入的文件中的变量$name将失效)。

#!/usr/bin/env perl
#
use strict;
use warnings;
use 5.010; {
do 'sum.pm';
die $@ if $@;
my $sum=sum(1,2,3);
say $sum;
# say $name; # 因为use strict的存在,而报错
}
say sum(2,3,4); # 出了语句块,函数仍有效

注意,do语句是在当前文件中引入变异结果,而不是当前代码块。

do导入文件时如果使用的是相对路径(如do 'sum1.pm'),将搜索@INC,搜索到后将更新%INC保持跟踪。

require导入文件

想象一下,如果在myperl.pm中使用do一次性导入两个文件sum1.pm、sum2.pm:

do 'sum1.pm';
die $@ if $@;
do 'sum2.pm';
die $@ if $@;

假设sum2.pm中也用了do语句导入sum1.pm,这样将会在myperl.pm中多次导入sum1.pm。其实第二次导入是多余的,尽管两次导入的内容是完全一致的,而且如果开启了use warnings,将会发出警告。

使用require语句可以解决这样重定义问题。

require '/perlapp/sum1.pm';   # 要给定正确的路径
require '/perlapp/sum2.pm';

require会在hash结构%INC中跟踪已经成功导入的文件,即使sum2.pm中也有require 'sum1.pm'语句。

为了跟踪是否曾经导入成功,要求所导入的文件最后要返回一个true,一般都会使用数值1作为所导入文件代码的结尾。并非一定是1,也可以是其它值,只要能表示最后这个文件是成功的就行。

例如,sum1.pm中:

#!/usr/bin/env perl
use strict;
use warnings;
use 5.010; sub sum {
my $sum;
$sum = map { $_ + $sum } $@
$name;
}
1; # 最后一行用1表示成功

以下是require相关的几点特性:

  • require的本质上do语句,do语句的本质是eval
  • require是在程序运行时执行的
  • 所导入文件中的任何语法错误都回直接die,因此可以省略die $@ if $@
  • require还能用于要求版本号,例如:require 5.010;
  • 在使用require时,如果使用的是裸词,例如require Foo::Bar;,将搜索@INC中的Foo/Bar.pm文件
  • 如果使用的不是裸词,如下。如果是绝对路径,则按照绝对路径查找,如果是相对路径,将从@INC路径下直接搜索Foo::Bar文件,显然文件一般不会这样命名,将会发出警告
    • require "Foo::Bar";:(双引号的存在)
    • require $class;其中$class="Foo::Bar"
    • require 'myperl.pm';

Perl导入代码文件的更多相关文章

  1. Java解析导入Excel文件后台代码实现

    使用MultipartFile上传Excel文件后端代码实现:(springmvc下的spring-webmvc (MultipartFile )上传) 由于POST一个包含文件上传的Form会以mu ...

  2. Spring Boot下的一种导入Excel文件的代码框架

    1.前言 ​ Spring Boot下如果只是导入一个简单的Excel文件,是容易的.网上类似的文章不少,有的针对具体的实体类,代码可重用性不高:有的利用反射机制或自定义注解,开发了Excel导入工具 ...

  3. mysql使用load导入txt文件所遇到的问题及解决方法

    导入txt文件,有导入向导这种方式: 另外可以使用load的方式导入.最开始使用以下代码插入: load data local infile 'F:\\Data\\predict_data.txt' ...

  4. Linux环境导入*.sql文件出现数据库为空

    登录mysql命令: 导入.sql文件: 或者: mysql -h127.0.0.1 -uroot -p userDb < /home/user.sql  按回车键后输数据库的密码 导入成功后, ...

  5. php通用安装程序,导入数据文件(.sql)的安装程序

    php通用安装程序,导入数据文件(.sql)的安装程序 该程序只需要1个php文件 和 1个数据文件,很方便调用.install/index.php         程序文件install/mycms ...

  6. Eclipse在已创建的project中导入其他文件

    Eclipse在已创建的project中导入其他文件 前两天被同事问到,如何通过不拷贝源文件的方式,在之前已经创建好的project中直接导入其他目录下的文件, 整理了一下,将目前所知道的eclips ...

  7. Java POI导入Excel文件

    今天在公司需要做个导入Excel文件的功能,所以研究了一下,参考网上的一些资料总算是做出来了,在此记录一下防止以后忘记怎么弄. 本人用的是poi3.8,所以需要的JAR包如下: poi-3.8.jar ...

  8. geotools导入shp文件到Oracle数据库时表名带下划线的问题解决

    问题: 最近在做利用geotools导入shp文件到Oracle表中,发现一个问题Oracle表名带下划线时导入失败,问题代码行: dsOracle.getFeatureWriterAppend(or ...

  9. 如何用CSC.exe来编译Visual C#的代码文件

    原文:如何用CSC.exe来编译Visual C#的代码文件 C#的编译过程      如何用CSC.exe来编译Visual C#的代码文件   Csc.exe 编译器的位置路径:C:\Window ...

随机推荐

  1. Python 协程 61

    什么是协程 协程,又称微线程,纤程.英文名Coroutine.一句话说明什么是线程:协程是一种用户态的轻量级线程. 协程的特点 协程拥有自己的寄存器上下文和栈.协程调度切换时,将寄存器上下文和栈保存到 ...

  2. 《NoSQL精粹》读后感

    <NoSQL精粹>作者Pramod J. Sadalaga.Martin Flower著,译者爱飞翔. 本书以关系型数据库开头,讲解了关系型数据库的优缺点,然后引入了NoSQL数据库,并且 ...

  3. HDU - 1241 Oil Deposits 经典dfs 格子

    最水的一道石油竟然改了一个小时,好菜好菜. x<=r  y<=c  x<=r  y<=c  x<=r  y<=c  x<=r y<=c #include ...

  4. 学习Python第七天

    进制拾遗: 二进制:01 八进制:01234567 十进制:0123456789 十六进制:0123456789ABCDEF 十进制转换八,十六进制语法 oct()八进制 关于8进制是逢8进一位的,我 ...

  5. 关于NGUI Shader 和 Draw Call的优化 & 模糊shader

    序: 1.项目过程中不可避免的需要用到大量Shader 和 UITexture,由于Ngui对Shader支持非常糟糕,导致项目drawCall异常的高 2.Panel裁剪无法裁剪自定义shader内 ...

  6. day_3各种数据类型与各种运算符

    首先我们复习一下昨天的内容 1:语言的分类: --有三种 机器语言,汇编语言,高级语言 运行的效率是机器语言最高  开发效率 是高级语言最高 2:计算机由五大部分组成:控制器+运算器+存储器+inpu ...

  7. Redisson碰到的问题

    最近开发环境使用redisson(版本是2.8.0),在部署一段时间(半个小时左右),获取分布式锁会报超时异常(org.redisson.client.RedisTimeoutException: R ...

  8. Monad Explained in One Picture

    The point of Monad is composability. In the green category, T -> Monad<U> and U -> Monad ...

  9. php安全编程&python测试实例编写

    前言 本文首发i春秋论坛. 本篇文章主要分享一个python暴破脚本,该脚本采用optparse模块,支持自定义字典位置:用了多线程(虽然我感觉和单线程速度差不多..是我的错觉还是线程写的不对..求表 ...

  10. C/C++ 多线程机制

    一.C/C++多线程操作说明 C/C++多线程基本操作如下: 1. 线程的建立结束 2. 线程的互斥和同步 3. 使用信号量控制线程 4. 线程的基本属性配置 在C/C++代码编写时,使用多线程机制, ...