Moose-0.92 > Moose::Cookbook::Basics::Recipe9

題名

Moose::Cookbook::Basics::Recipe9 - 演算子のオーバーロード、サブタイプ、型変換

概要

  package Human;

  use Moose;
  use Moose::Util::TypeConstraints;

  subtype 'Gender'
      => as 'Str'
      => where { $_ =~ m{^[mf]$}s };

  has 'gender' => ( is => 'ro', isa => 'Gender', required => 1 );

  has 'mother' => ( is => 'ro', isa => 'Human' );
  has 'father' => ( is => 'ro', isa => 'Human' );

  use overload '+' => \&_overload_add, fallback => 1;

  sub _overload_add {
      my ( $one, $two ) = @_;

      die('Only male and female humans may create children')
          if ( $one->gender() eq $two->gender() );

      my ( $mother, $father )
          = ( $one->gender eq 'f' ? ( $one, $two ) : ( $two, $one ) );

      my $gender = 'f';
      $gender = 'm' if ( rand() >= 0.5 );

      return Human->new(
          gender => $gender,
          mother => $mother,
          father => $father,
      );
  }

本文

このレシピでは演算子のオーバーロードや型変換、サブタイプを利用してヒトの繁殖システムを(まあ、少なくとも遺伝子の選択くらいは、ですが)再現する方法を紹介します。

はじめに

このHumanクラスでは演算子のオーバーロードを利用して2人のヒトを「加え」れば子供が産まれるようになっています(今回の実装では2つのオブジェクトは異なる性を持っている必要がありますが、ここで話題にしているのはあくまでも生物学的な繁殖の話であって、結婚の話ではありませんので、あしからず)。

この例は、そのままでもよいのですが、遺伝子を加えてやればもっとおもしろいものになります。ここでは生物学のまねごとをするために目の色をコントロールする遺伝子を2つ加えて、オーバーロードを使って親から受け継いだ遺伝子を合成してみます。

演算子のオーバーロードとは?

オーバーロードはMoose特有の機能「ではありません」。Perlではoverloadプラグマを使って実装されている一般的なオブジェクト指向のコンセプトです。オーバーロードを使うと、加算演算子(+)のようにPerlの組み込み演算子とオブジェクトを組み合わせて使うとき、あるいはオブジェクトを文字列として扱うときに有意義な処理をさせられます。

この例では、加算をオーバーロードして、$child = $mother + $fatherのようなコードを書けるようにしています。

遺伝子

目の色に影響を与える遺伝子はたくさんありますが、もっとも重要なのはgeybey2の2つです。ここではそれぞれの遺伝子にクラスを用意するところから始めましょう。

Human::Gene::bey2

  package Human::Gene::bey2;

  use Moose;
  use Moose::Util::TypeConstraints;

  type 'bey2_color' => where { $_ =~ m{^(?:brown|blue)$} };

  has 'color' => ( is => 'ro', isa => 'bey2_color' );

このクラスはたいしたことはありません。許される色についての型制約と、colorというアトリビュートがあるだけです。

Human::Gene::gey

  package Human::Gene::gey;

  use Moose;
  use Moose::Util::TypeConstraints;

  type 'gey_color' => where { $_ =~ m{^(?:green|blue)$} };

  has 'color' => ( is => 'ro', isa => 'gey_color' );

これもHumane::Gene::bey2クラスとほとんど同じですが、許される色が異なっています。

目の色

Humanクラスに(それぞれの遺伝子について2つずつ)4つのアトリビュートを用意することもできますが、それではいささかごちゃごちゃしてしまいますので、かわりに遺伝子をHuman::EyeColorというコンテナクラスに入れて抽象化することにします。こうすると、Humanにはeye_colorというアトリビュートさえ用意しておけばよくなります。

  package Human::EyeColor;

  use Moose;
  use Moose::Util::TypeConstraints;

  coerce 'Human::Gene::bey2'
      => from 'Str'
          => via { Human::Gene::bey2->new( color => $_ ) };

  coerce 'Human::Gene::gey'
      => from 'Str'
          => via { Human::Gene::gey->new( color => $_ ) };

  has [qw( bey2_1 bey2_2 )] =>
      ( is => 'ro', isa => 'Human::Gene::bey2', coerce => 1 );

  has [qw( gey_1 gey_2 )] =>
      ( is => 'ro', isa => 'Human::Gene::gey', coerce => 1 );

目の色をあらわすクラスには遺伝子がそれぞれ2つずつ用意されています。また、それぞれのクラスの型変換も用意して、文字列を新しいオブジェクトに変換できるようにしてあります。ただし、「indigo」のような文字列を変換しようとすると、型変換は失敗します。どちらの遺伝子もこの色を有効とは認めないためです。

ついでながら、この例を見ると、hasの第1引数に名前の配列リファレンスを渡すと、同じアトリビュートを一度に複数個定義できるのがわかります。

それから、遺伝子の組み合わせから実際の目の色を計算するメソッドが必要です。bey2の茶色の遺伝子は青と緑の遺伝子より優性で、geyの緑の遺伝子は青より優性です。

  sub color {
      my ($self) = @_;

      return 'brown'
          if ( $self->bey2_1->color() eq 'brown'
          or $self->bey2_2->color() eq 'brown' );

      return 'green'
          if ( $self->gey_1->color() eq 'green'
          or $self->gey_2->color() eq 'green' );

      return 'blue';
  }

Human::EyeColorオブジェクトは文字列としても扱えるようにしたいので、Human::EyeColorクラスに文字列のオーバーロードを定義しましょう。

  use overload '""' => \&color, fallback => 1;

最後に、加算のオーバーロードを定義する必要があります。これで、Human::EyeColorオブジェクトを足しあわせると、新しい(遺伝子学的に正しい)目の色をした新しいHuman::EyeColorオブジェクトが得られるようになります。

  use overload '+' => \&_overload_add, fallback => 1;

  sub _overload_add {
      my ( $one, $two ) = @_;

      my $one_bey2 = 'bey2_' . _rand2();
      my $two_bey2 = 'bey2_' . _rand2();

      my $one_gey = 'gey_' . _rand2();
      my $two_gey = 'gey_' . _rand2();

      return Human::EyeColor->new(
          bey2_1 => $one->$one_bey2->color(),
          bey2_2 => $two->$two_bey2->color(),
          gey_1  => $one->$one_gey->color(),
          gey_2  => $two->$two_gey->color(),
      );
  }

  sub _rand2 {
      return 1 + int( rand(2) );
  }

目の色をあらわすオブジェクトを2つ足しあわせると、_overload_add()メソッドに2つのHuman::EyeColorオブジェクトが渡され(+演算子の左項と右項です)、新しいHuman::EyeColorオブジェクトが返ります。

Humanに目の色を追加する

新しいHuman::EyeColorクラスは、もとのHumanクラスを何ヶ所か変更するだけで組み込むことができます。

  use List::MoreUtils qw( zip );

  coerce 'Human::EyeColor'
      => from 'ArrayRef'
      => via { my @genes = qw( bey2_1 bey2_2 gey_1 gey_2 );
               return Human::EyeColor->new( zip( @genes, @{$_} ) ); };

  has 'eye_color' => (
      is       => 'ro',
      isa      => 'Human::EyeColor',
      coerce   => 1,
      required => 1,
  );

また、Humanクラスの_overload_add()も目の色に対応できるように修正する必要があります。

  return Human->new(
      gender    => $gender,
      eye_color => ( $one->eye_color() + $two->eye_color() ),
      mother    => $mother,
      father    => $father,
  );

まとめ

ここで利用したオーバーロード、サブタイプ、型変換という3つのテクニックを組み合わせると強力なインタフェースを提供できます。

オーバーロードの詳細についてはoverloadプラグマのドキュメントをご覧ください。

ここで作成したコードの全体をまとめて見たい場合はt/000_recipes/basics/010_genes.tをご覧ください。

次のステップ

これが本物のプロジェクトだったとしたら、おそらく以下のようなものがほしくなるでしょう。

Crypt::Randomを使った高度なランダム化
身体的特徴用のベースクラス
遺伝子の突然変異
追加の身体的特徴
人工生命

作者

Aran Clary Deltac <bluefeet@cpan.org>

Dave Rolsky <autarch@urth.org>

ライセンス

This work is licensed under a Creative Commons Attribution 3.0 Unported License.

License details are at: http://creativecommons.org/licenses/by/3.0/