Perl`defined’和`undef’子例程范围

请看下面的代码
use strict;
use warnings;


print "subroutine is defined\n" if defined &myf;
myf();

sub myf
{
        print "called myf\n";
}

undef &myf;
#myf();
print "Now subroutine is defined\n" if defined &myf;

输出

subroutine is defined
called myf

一个print语句可以打印,这是否意味着解释器(或编译器?)进一步查看并查看子例程定义?如果是这样,为什么它没有看到undef& myf;作为第二个印刷声明?

谢谢

解决方法

这与范围无关,但与编译时和运行时有关.这是一个简化的解释.

Perl解释器最初将扫描您的代码,并遵循任何使用语句或BEGIN块.此时,它会看到所有潜艇,并在各自的包装中记下它们.所以现在你的符号表中有一个& :: myf.

当编译时间到达程序结束时,它将切换到运行时.

那时,它实际上运行代码.如果定义了& myf,则执行第一个print语句.我们知道它是,因为它在编译时设置.然后Perl调用函数.一切都很好.现在你在符号表中取消该条目.这也发生在运行时.

之后,定义的& myf返回false,因此不会打印.

你甚至在代码中第二次调用myf(),但是注释掉了.如果你删除评论,它会抱怨未定义的子程序& main :: myf被调用.这对发生的事情有很好的暗示.

所以实际上它在代码中没有向前或向后看.它已经在那时完成了扫描代码.

in perlmod解释了不同的阶段.

请注意,实际上没有使用函数的用例并不多.除非您想手动清理命名空间,否则我不明白为什么要删除它.

相关文章

1. 如何去重 #!/usr/bin/perl use strict; my %hash; while(...
最近写了一个perl脚本,实现的功能是将表格中其中两列的数据...
表的数据字典格式如下:如果手动写MySQL建表语句,确认麻烦,...
巡检类工作经常会出具日报,最近在原有日报的基础上又新增了...
在实际生产环境中,常常需要从后台日志中截取报文,报文的形...
最近写的一个perl程序,通过关键词匹配统计其出现的频率,让...