长话短说:我们想标记字符串,以便以后即使它们嵌入其他字符串中,我们也可以对其进行处理。
所以我们想到了,嘿,让我们尝试重载。这很整洁。我可以做类似的事情:
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;
我希望所有这些都作为对象而不是字符串转储。但是“坏”的例子都被串了。所有“ BAD”示例都是当我将当前要连接的字符串对象分配给先前声明的变量时。如果我同时声明,或者以前连接了字符串,或者添加了额外的串联(除了插值字符串concat之外),那么它就可以正常工作。
这太疯狂了。
脚本的结果:
$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' )
]
};
这种行为对我来说毫无意义。我想了解为什么它会这样工作,所以我想找到一种解决方法。
答案 0 :(得分:2)
这不是一个很好的解决方案,也无法回答为什么perl会这样做,但是我得到了一些东西……我在其中留下了一些调试打印语句。
无论出于何种原因,perl都认为您希望将对对象的标量引用转换为标量字符串。您可以通过添加对引用的引用,然后再对其取消引用来欺骗其不执行此操作。
RewriteRule ^([a-z]+)/([A-Za-z0-9\%]+)/?$ index.php?get1=$1&get2=$2 [NC,L,B]
$替代方法可为您提供以下信息
#!/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;