长话短说:我们想要标记字符串,这样以后我们就可以对它们做一些事情,即使它们被嵌入到其他字符串中。
所以我们想,嘿,让我们试试重载。它相当整洁。我可以这样做:
my $str = str::new('<encode this later>');
my $html = "<html>$str</html>";
print $html; # <html><encode this later></html>
print $html->encode; # <html><encode this later></html>它通过重载连接运算符来创建一个新的对象数组,其中包含普通字符串"“、对象换行"”和普通字符串"“。它可以任意地嵌套这些。在编码时,它将保留普通字符串,但对对象字符串进行编码。但是如果你把对象串起来,它只会把它变成简单的字符串。
这工作得很好,除了在某些情况下,它没有明显的原因而串行化。下面的脚本显示了我在5.10到5.22中复制的行为。
#!/usr/bin/perl
use strict;
use warnings;
use 5.010;
use Data::Dumper; $Data::Dumper::Sortkeys=1;
my $str1 = str::new('foo');
my $str2 = str::new('bar');
my $good1 = "$str1 $str2";
my $good2;
$good2 = $good1;
my($good3, $good4);
$good3 = "$str1 a";
$good4 = "a $str1";
my($bad1, $bad2, $bad3);
$bad1 = "a $str1 a";
$bad2 = "$str1 $str2";
$bad3 = "a $str1 a $str2 a";
say Dumper { GOOD => [$good1, $good2, $good3], BAD => [$bad1, $bad2, $bad3] };
$bad1 = ''."a $str1 a";
$bad2 = ''."$str1 $str2";
$bad3 = ''."a $str1 a $str2 a";
say Dumper { BAD_GOOD => [$bad1, $bad2, $bad3] };
package str;
use Data::Dumper; $Data::Dumper::Sortkeys=1;
use strict;
use warnings;
use 5.010;
use Scalar::Util 'reftype';
use overload (
'""' => \&stringify,
'.' => \&concat,
);
sub new {
my($value) = @_;
bless((ref $value ? $value : \$value), __PACKAGE__);
}
sub stringify {
my($str) = @_;
#say Dumper { stringify => \@_ };
if (reftype($str) eq 'ARRAY') {
return join '', @$str;
}
else {
$$str;
}
}
sub concat {
my($s1, $s2, $inverted) = @_;
#say Dumper { concat => \@_ };
return new( $inverted ? [$s2, $s1] : [$s1, $s2] );
}
1;我希望所有这些都作为对象而不是字符串来转储。但是“坏”的例子都是字符串形式的。所有“坏”的例子都是当我将一个string对象赋值给一个先前声明的变量时。如果我同时声明,或者之前连接字符串,或者添加额外的连接(在插入的字符串连接之外),那么它工作得很好。
这太疯狂了。
脚本的结果:
$VAR1 = {
'BAD' => [
'a foo a',
'foo bar',
'a foo a bar a'
],
'GOOD' => [
bless( [
bless( [
bless( do{\(my $o = 'foo')}, 'str' ),
' '
], 'str' ),
bless( do{\(my $o = 'bar')}, 'str' )
], 'str' ),
$VAR1->{'GOOD'}[0],
bless( [
$VAR1->{'GOOD'}[0][0][0],
' a'
], 'str' )
]
};
$VAR1 = {
'BAD_GOOD' => [
bless( [
'',
bless( [
bless( [
'a ',
bless( do{\(my $o = 'foo')}, 'str' )
], 'str' ),
' a'
], 'str' )
], 'str' ),
bless( [
'',
bless( [
bless( [
$VAR1->{'BAD_GOOD'}[0][1][0][1],
' '
], 'str' ),
bless( do{\(my $o = 'bar')}, 'str' )
], 'str' )
], 'str' ),
bless( [
'',
bless( [
bless( [
bless( [
bless( [
'a ',
$VAR1->{'BAD_GOOD'}[0][1][0][1]
], 'str' ),
' a '
], 'str' ),
$VAR1->{'BAD_GOOD'}[1][1][1]
], 'str' ),
' a'
], 'str' )
], 'str' )
]
};这种行为对我来说毫无意义。我想要了解为什么它是这样工作的,并且我想找到一种解决方法。
发布于 2018-08-17 12:07:08
嗯,这不是一个很好的解决方案,也不能解释为什么perl要这样做,但我得到了一些东西……我在其中留下了一些调试print语句。
无论出于什么原因,perl都认为您希望将对象的标量引用转换为标量字符串。您可以通过添加对引用的引用,然后取消对它的引用来诱使它不这样做。
#!/usr/bin/perl
use strict;
use warnings;
use 5.010;
use Data::Dumper; $Data::Dumper::Sortkeys=1;
use Scalar::Util 'reftype';
my $str1 = str::new('foo');
my $str2 = str::new('bar');
say 'good1';
my $good1 = "$str1 $str2";
say 'g1 ', reftype($good1);
say Dumper $good1;
say 'bad1';
my $bad1;
say 'b1 ', reftype($bad1);
$bad1 = "$str1 $str2";
say 'b2 ', reftype($bad1);
say Dumper $bad1;
say 'workaround';
my $workaround;
say 'w1 ', reftype($workaround);
$workaround = ${\"$str1 $str2"};
say 'w2 ', reftype($workaround);
say Dumper $workaround;
package str;
use Data::Dumper; $Data::Dumper::Sortkeys=1;
use strict;
use warnings;
use 5.010;
use Scalar::Util 'reftype';
use overload (
'""' => \&stringify,
'.' => \&concat,
);
sub new {
my ($value) = @_;
bless((ref $value ? $value : \$value), __PACKAGE__);
}
sub stringify {
my ($str) = @_;
say "stringify";
say reftype($str);
if (reftype($str) eq 'ARRAY') {
say scalar @$str;
return join '', @$str;
}
else {
$$str;
}
}
sub concat {
my ($s1, $s2, $inverted) = @_;
say "concat";
say reftype($s1);
say reftype($s2);
say reftype($inverted);
return new( $inverted ? [$s2, $s1] : [$s1, $s2] );
}
1;$workaround为您提供了以下内容
$VAR1 = bless( [
bless( [
bless( do{\(my $o = 'foo')}, 'str' ),
' '
], 'str' ),
bless( do{\(my $o = 'bar')}, 'str' )
], 'str' );https://stackoverflow.com/questions/50998365
复制相似问题