我不喜欢没有对象化的深层对象,幸运的是,Moose 内置了强制功能,因此您几乎可以像魔术一样对象化深层对象。
我有点过火了,但我决定继续并把它记下来作为自己的练习,尽管我认为我可以“扮演”一些项目并获得更好的结果,或者强制强制 Alpha::Keyed to无论如何,从必填字段构建结果类。
我不完全喜欢我编码的方式,但我不想在上面花费大量时间,但它适用于您上面的对象。您必须做很多工作才能使其在更复杂的对象上运行,并且您需要将代码分解为单独的类:
阿尔法下午:
package Alpha;
use Moose;
use Moose::Util::TypeConstraints;
subtype 'AlphaCodes',
as 'Alpha::Codes';
subtype 'AlphaData',
as 'Alpha::Data';
coerce 'AlphaCodes',
from 'ArrayRef[HashRef]',
via { Alpha::Codes->new( data => $_ ) };
coerce 'AlphaData',
from 'HashRef',
via { Alpha::Data->new($_) };
has 'code' => (
is => 'ro',
isa => 'AlphaCodes',
required => 1,
coerce => 1);
has 'data' => (
is => 'ro',
isa => 'AlphaData',
required => 1,
coerce => 1);
package Alpha::Codes;
use Moose;
use Moose::Util::TypeConstraints;
extends 'Alpha::KeyedList';
subtype 'ArrayRefOfCodes',
as 'ArrayRef[Alpha::Code]';
coerce 'ArrayRefOfCodes',
from 'ArrayRef[HashRef]',
via { [ map { Alpha::Code->new($_) } @$_ ] };
has 'data' => (
is => 'ro',
isa => 'ArrayRefOfCodes',
required => 1,
coerce => 1);
package Alpha::KeyedList;
use Moose;
use Moose::Util::TypeConstraints;
sub unique_list {
my $self = shift;
my %seen = ();
my @retval = ();
foreach my $item ( @{$self->data} ) {
unless ( $seen{$item->key} ) {
push(@retval,$item);
$seen{$item->key} = 1;
}
}
return @retval;
}
package Alpha::Data;
use Moose;
use Moose::Util::TypeConstraints;
subtype 'AlphaDataDomestics',
as 'Alpha::Data::Domestics';
coerce 'AlphaDataDomestics',
from 'ArrayRef[HashRef]',
via { Alpha::Data::Domestics->new(data => $_) };
has 'domestic' => (
is => 'ro',
isa => 'AlphaDataDomestics',
required => 1,
coerce => 1 );
package Alpha::Data::Domestics;
use Moose;
use Moose::Util::TypeConstraints;
extends 'Alpha::KeyedList';
subtype 'ArrayRefOfDomestics',
as 'ArrayRef[Alpha::Data::Domestic]';
coerce 'ArrayRefOfDomestics',
from 'ArrayRef[HashRef]',
via { [ map { Alpha::Data::Domestic->new($_) } @$_ ] };
has 'data' => (
is => 'ro',
isa => 'ArrayRefOfDomestics',
required => 1,
coerce => 1);
package Alpha::Data::Domestic;
use Moose;
extends 'Alpha::Keyed';
has 'a' => ( is => 'ro' , isa => 'Str' , required => 1 );
has 'b' => ( is => 'ro' , isa => 'Str' , required => 1 );
sub build_key {
my $self= shift;
return $self->a . '__' . $self->b;
}
package Alpha::Code;
use Moose;
extends 'Alpha::Keyed';
has 'x' => ( is => 'ro' , isa => 'Str' , required => 1 );
has 'y' => ( is => 'ro' , isa => 'Str' , required => 1 );
sub build_key {
my $self= shift;
return $self->x . '__' . $self->y;
}
package Alpha::Keyed;
use Moose;
has 'key' => ( is => 'ro'
, isa => 'Str'
, builder => 'build_key'
, lazy => 1 );
package main;
my $VAR1 = {
'alpha' => {
'code' => [
{
'x' => 1,
'y' => 2
},
{
'x' => 1,
'y' => 2
}
],
'data' => {
'domestic' => [
{
'a' => 0,
'b' => 5
},
{
'a' => 0,
'b' => 5
},
{
'a' => 1,
'b' => 2
},
]
}
}
};
my $alpha = Alpha->new($VAR1->{alpha});
use Data::Dumper;
warn Dumper([ $alpha->code->unique_list ]);
warn Dumper([ $alpha->data->domestic->unique_list ]);
1;
现在运行:
$VAR1 = [
bless( {
'y' => 2,
'x' => 1,
'key' => '1__2'
}, 'Alpha::Code' )
];
$VAR1 = [
bless( {
'a' => 0,
'b' => 5,
'key' => '0__5'
}, 'Alpha::Data::Domestic' ),
bless( {
'a' => 1,
'b' => 2,
'key' => '1__2'
}, 'Alpha::Data::Domestic' )
];