如何在 Perl 中添加用户定义函数作为内置函数?

asp*_*dev 1 oop perl code-reuse reusability

有一次,我在考虑如何在 Perl 中创建一个包(或类)以实现代码的可重用性。我有 C++ 背景,无法理解 Perl 子例程和包的工作原理。所以,我在 Perl 中创建了一个 Stack 数据结构(它实际上只是一个具有 LIFO 方法的数组),每次我想打印它时都需要我写下循环(这是非常低效的)

@_ = (1, 8, 14, 45);

push(@_, 69);
push(@_, 55);
push(@_, 65536);
push(@_, 4294967296);

print "Print without newline: ";

$size = @_;
for my $i(1 .. ($size-1))
{
    print $_[$i]." ";
}

print "Print with newline: ";

for my $i(1 .. ($size-1))
{
    say scalar $_[$i];
}

pop(@_);
pop(@_);
pop(@_);

print "Print without newline: ";
# HERE WE GO AGAIN
$size = @_;
for my $i(1 .. ($size-1))
{
    print $_[$i]." ";
}
Run Code Online (Sandbox Code Playgroud)

然后,我开始学习更多关于创建包以重用代码的知识。这是我做的一个例子。

打印机.pm

use Modern::Perl;

package Printer;
sub new
{
    my $class = shift;
    my $self = {};

    bless $self, $class;

    return $self;
}

my $size;
sub println
{   
    my @userArray = @_;
    $size = @userArray;
    for my $i(1 .. ($size-1))
    {
        say scalar $userArray[$i];
    }
}

sub print
{   
    my @userArray = @_;
    $size = @userArray;
    for my $i(1 .. ($size-1))
    {
        print $userArray[$i]." ";
    }

}

1;
Run Code Online (Sandbox Code Playgroud)

Stack.pl 的伪代码

use lib ".\\";
use Array::Printer;

. . .

my $printer = new Printer();
print "Print without newline: ";
print $printer->print(@_);
print "\nPrint with newline: \n";
print $printer->println(@_);

say "";

# POPPING (LIFO)
pop(@_);
pop(@_);
pop(@_);

print $printer->print(@_);
Run Code Online (Sandbox Code Playgroud)

而且,正如你所猜到的,它就像魅力一样。

然而有一天,一件事突然出现在我的脑海中。我想知道是否可以将我在 Printer 包中内置的“println”函数作为“内置”函数。没有调用包的构造函数。我在想这样的事情:

println(@_);
Run Code Online (Sandbox Code Playgroud)

由于 println 在 Perl 中不存在,我只是好奇是否有可能这样做。关于我接下来应该做什么的任何建议?

Dav*_*oss 6

如果你想模拟一个内置的,那么你不需要一个类,你只需要一个库。库也在 Perl 中实现为包,但其中的子例程不希望将类名或对象作为它们的第一个参数传递。您还可以使用Exporter将您的子程序插入到调用程序的符号表中。

它看起来像这样:

package Array::Printer;

use strict;
use warnings;
use feature 'say';

require Exporter;
our @ISA = qw[Exporter];
our @EXPORT = qw[println];

sub println {
  say for @_;
}

1;
Run Code Online (Sandbox Code Playgroud)

你会像这样使用它:

#!/usr/bin/perl

use strict;
use warnings;

use lib '.';
use Array::Printer;

# I've used @stack instead of @_ in my code as @_ is a
# special variable and it seems strange to use it in another way.

my @stack = (1, 8, 14, 45);
push(@stack, 69);
push(@stack, 55);
push(@stack, 65536);
push(@stack, 4294967296);

println(@stack);

pop(@stack);
pop(@stack);
pop(@stack);

println(@stack);
Run Code Online (Sandbox Code Playgroud)

更新:请注意,我的库模块名为Array::Printer. 您的原始类应该使用相同的名称。通常,Perl 包名称(库和类)应该与存储它们的文件的名称相匹配。您调用了包,Printer但使用了Array::Printer. 这肯定会让任何维护程序员(甚至你!)

use lib '.'在我的代码将当前目录的目录列表是Perl搜索类和库。因此,我们假设 for 的代码Array::Printer在一个文件Printer.pm中,该文件Array位于您当前目录中的一个目录中(该目录大概也是您的主程序所在的位置。Perl 库名称中的双冒号 ( ::) 被转换为/查找包含库的文件时的目录分隔符 ( )。