我不完全理解the docs,所以我尝试了clone
,似乎有一个可变类的属性,它可以在旧对象中使用旧的更改一个(这就是我不想要的)。如何使它们(即副本和原件)完全分开?
class A {
has @.a;
}
my A $x = A.new;
my A $y = A.new;
$x.a = 1, 2;
$y = $x.clone;
$x.a.push(4);
say $y.a; # [1 2 4]
答案 0 :(得分:9)
从clone
继承的默认Mu
很浅,如文档所述。这意味着它只会复制对象本身,而不会复制对象引用的任何内容。可以覆盖clone
以获得您的首选语义,这可能是最好的事情。
执行此操作时需要注意的一点是clone
获取命名参数并使用它们来分配克隆对象的属性。这值得了解,因为:
clone
时处理此问题,以避免使用此功能的被覆盖clone
方法的用户感到意外clone
以简洁地选择克隆特定数组或哈希属性时,可以使用此方法因此对于问题中的案例,写作:
class A {
has @.a;
method clone() {
callwith(:@!a, |%_)
}
}
可能会导致输出[1 2]
。它是如何工作的?
clone
方法:@!a
是a => @!a
callwith
调用继承的clone
(在这种情况下来自Mu
)@!a
上使用赋值,而不是绑定,语义(就像在对象构造期间一样),从而生成数组的副本此快捷方式也适用于哈希属性。对于包含其他对象的属性,它看起来像callsame(x => $!x.clone)
。