如何在Perl 6中重新生成对象?

时间:2017-06-11 17:56:38

标签: inheritance perl6

另一个问题可能是"我如何从内置类型继承?"。

我确实有两个问题,但它们都与我玩的一样。

首先,当我想进一步约束它时,我可以创建一个类型的子集。我使用MyInt来接受Int的任何内容。我通过MyInt声明了一个变量并分配给它,但是当我检查它的名字时,我会得到Int。那么,那是什么呢?

subset MyInt where * ~~ Int;

my MyInt $b = 137;
put 'Name: ', $b.^name;  # Int, but why not MyInt?

但是,我真正想要的是一个名为MyInt的类,它做同样的事情。我可能想要添加方法

class MyInt is Int {}   # empty subclass

my MyInt $b = 137;
put 'Name: ', $b.^name;

这几乎看起来很有效,但我收到了一个错误:

Type check failed in assignment to $b; expected MyInt but got Int (137)

我理解它的含义,但不明白为什么我在使用subset时没有遇到同样的错误。那个问题1.5。

我真正喜欢的是分配137以在分配时将其自动转换为MyInt。我知道我可以明确地构造它,但是有点烦人的是,父类仍然把它变成Int而不是使用更派生类型的类型:

class MyInt is Int {}   # empty subclass

my MyInt $b = MyInt.new: 137;  # Still an Int
put 'Name: ', $b.^name;

我可以覆盖new(直接来自Int.pm),但我不知道改变类型:

class MyInt is Int {
    method new ( $value --> MyInt ) {
        my $v = callsame; # let superclass construct it
        # But, how do I make it the more specific type?
        }
    }

my MyInt $b = MyInt.new: 137;  # Still an Int
put 'Name: ', $b.^name;

我可以bless自我,但这并不能保留价值(而且我认为它不会,也不会认为应该这样。看Int.pm ,我无法看到它如何存储该值。它看起来依赖于内置类型,可能不是传统的子类:

class MyInt is Int {
    method new ( $value --> MyInt ) {
        my $v = callsame; # let superclass construct it
        put "v is $v";
        # But, how do I make it the more specific type?
        # $v.bless (doesn't change the type, fails return type check)
        self.bless;  # doesn't retain value
        }
    }

my MyInt $b = MyInt.new: 137;  # Still an Int
put 'Name: ', $b.^name;
put 'Value: ', $b;  # 0

有一个rebless,但这不是IntClassHow可以链接的事物的一部分:

class MyInt is Int {
    method new ( $value --> MyInt ) {
        my $v = callsame; # let superclass construct it
        put "v is $v";
        put "self is " ~ self.^name;
        put "HOW is " ~ self.HOW.^name;
        # No such method 'rebless' for invocant
        # $v.rebless: self.^name;
        $v.HOW.rebless: self.^name;
        }
    }

my MyInt $b = MyInt.new: 137;  # Still an Int
put 'Name: ', $b.^name;
put 'Value: ', $b;  # 0

1 个答案:

答案 0 :(得分:3)

这是一个可能的解决方案:

class MyInt is Int { };
my $x = 42;
Metamodel::Primitives.rebless: $x, MyInt;
dd $x;

产生:

MyInt $x = 42

这可能是一种更干净的方式来做你想做的事情,但我不知道它是什么。