The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.

NAME

Object::Simple::Guides::Ja - Object::Simpleの日本語ガイド

GUIDES

1. アクセサメソッドの生成

最初に、Object::Simpleのメソッドを利用するために、 Object::Simpleを継承したクラスを作成します

    package SomeClass;
    
    use base 'Object::Simple';

Object::Simplenew()メソッドを持っています。 これはコンストラクタで、 ハッシュかハッシュのリファレンスを受け取ることができます。

    my $obj = SomeClass->new;
    my $obj = SomeClass->new(foo => 1, bar => 2);
    my $obj = SomeClass->new({foo => 1, bar => 2});

attr()メソッドによって、アクセサメソッドを生成することができます。

    __PACKAGE__->attr('foo');

アクセサメソッドを使って値を設定したり取得することができます。

    # Set the value
    $obj->foo(1);
    
    # Get the value
    my $foo = $obj->foo;

アクセサメソッドのためのデフォルト値を指定することもできます。

    __PACKAGE__->attr(foo => 1);

fooの値が存在しないで、fooメソッドが呼び出されると、 デフォルト値を取得することができます。

    my $default_value = $obj->foo;

もしリファレンスやオブジェクトをデフォルト値として指定したい場合は デフォルト値を返却するサブルーチンのリファレンスである必要があります。 これはデフォルト値を他のオブジェクトと共有しないために必要です。

    __PACKAGE__->attr(foo => sub { [] });
    __PACKAGE__->attr(foo => sub { {} });
    __PACKAGE__->attr(foo => sub { SomeClass->new });

アクセサメソッドを一度に生成することもできます。

    __PACKAGE__->attr([qw/foo bar baz/]);
    __PACKAGE__->attr([qw/foo bar baz/] => 0);

サンプル:

Object::Simpleをよく理解するためにひとつのサンプルを見せます。

Pointクラスは、xyというふたつの属性と、 xyの値を0にクリアする<clear>というメソッドを持ちます。

    package Point;
    
    use strict;
    use warnings;
    
    use base 'Object::Simple';

    __PACKAGE__->attr(x => 0);
    __PACKAGE__->attr(y => 0);
    
    sub clear {
        my $self = shift;
        
        $self->x(0);
        $self->y(0);
    }

Point3Dクラスは、Pointクラスを継承します。 このクラスはxyという属性に加えて、zという属性 を持っています。 clear()メソッドはxyzの値をクリアするために オーバーライドされています。

    package Point3D;
    
    use strict;
    use warnings;
    
    use base 'Point';
    
    __PACKAGE__->attr(z => 0);
    
    sub clear {
        my $self = shift;
        
        $self->SUPER::clear();
        
        $self->z(0);
    }

2. オブジェクト指向プログラミングの概念

継承

Object::Simpleをよく理解するために、 オブジェクト指向の概念を解説したいと思います。

オブジェクト指向の一つ目の概念は「継承」です。 「継承」は「もしクラスQがクラスPを継承していたら、 クラスQはクラスPのすべてのメソッドを呼び出すことができる」 ということを意味します。

    +---+
    | P | Base class
    +---+   having method1() and method2()
      |
    +---+
    | Q | Sub class
    +---+   having method3()

クラスQはクラスPを継承しているので、 クラスQはクラスQのメソッドに加えて、クラスPのすべてのメソッド を呼び出すことができます。 言い換えれば、クラスQは method1(), method2()method3() を呼び出すことができます。

継承を行うには、baseモジュールを使用します。

    package P;
    
    sub method1 { ... }
    sub method2 { ... }
    
    package Q;
    
    use base 'P';
    
    sub method3 { ... }

Perlはオブジェクト指向プログラミングを助ける 便利な関数とメソッドを持っています。

オブジェクトがどのクラスに属しているかを知るには、 ref()関数を使用します。

    my $class = ref $obj;

オブジェクトが特定のクラスを継承しているかどうかを調べるには、 isa()メソッドを使用します。

    $obj->isa('SomeClass');

オブジェクト(あるいはクラス)が特定のメソッドを呼び出す ことができるかどうかを知るには、 can()メソッドを使用します。

    SomeClass->can('method1');
    $obj->can('method1');

カプセル化

オブジェクト指向プログラミングのふたつ目の概念は カプセル化です。 「カプセル化」は「内部的にデータに直接アクセスしてはいけない」 ということを意味します。 ドキュメントに記述された公開されたメソッドを使用しなければ なりません。 このルールを守ることによって、すべてのことがシンプルになります。

このルールを守るためには 値を取得や設定を行うためのアクセサメソッドを生成する 必要があります。

    my $value = $obj->foo;
    $obj->foo(1);

直接データにアクセスするのは良くない習慣です。

    my $value = $obj->{foo}; # Bad manner!
    $obj->{foo} = 1;         # Bad manner!

ポリモーフィズム

オブジェクト指向プログラミングの三つ目の概念は 「ポリモーフィズム」です。 「ポリモーフィズム」は、 「オーバーロード」と「オーバーライド」のふたつの概念に 分割されます。

Perlプログラマはオーバーロードを気にする必要はありません。 Perlは動的な言語なので、 サブルーチンはどのような値でも受け取ることができます。 オーバーロードはC++やJavaなどの 静的な型を持つ言語にとって価値があります。

「オーバーライド」は「サブクラスにおいて、基底クラスのメソッドを 変更することができる」ということを意味します。

    package P;
    
    sub method1 { return 1 }
    
    package Q;
    
    use base 'P';
    
    sub method1 { return 2 }

クラスPのmethod1()は1という値を返却します。 クラスQのmethod1()は2という値を返却します。 つまり、クラスQにおいて、method1()はオーバーライド されたということです。

    my $obj_a = P->new;
    $obj_p->method1; # Return value is 1
    
    my $obj_b = Q->new;
    $obj_q->method1; # Return value is 2

もし基底クラスのメソッドをサブクラスから呼び出したい場合は SUPER擬似クラスを使用します。

    package Q;
    
    sub method1 {
        my $self = shift;
        
        my $value = $self->SUPER::method1(); # return value is 1
        
        return 2 + $value;
    }

これらの三つの概念だけを理解するならば、 十分強力なオブジェクト指向プログラムができ、 ソースコードは他の言語のユーザから見ても 読みやすいものになるでしょう。

3. よく利用するテクニック

Override new()メソッドのオーバーライド

new()メソッドは必要ならばオーバーライドすることができます。

サンプル:

オブジェクトの初期化。

    sub new {
        my $self = shift->SUPER::new(@_);
        
        # Initialization
        
        return $self;
    }

サンプル:

new()の引数の変更。

    sub new {
        my $self = shift;
        
        $self->SUPER::new(x => $_[0], y => $_[1]);
        
        return $self;
    }
C<new()>メソッドをオーバーライドすることによって、
C<new()>に配列を渡すことができるようになりました。

    my $point = Point->new(4, 5);

4. その他の特徴

厳格な引数チェック

Object::Simpleはユーザビリティにも注意を払っています。 もし間違った数の引数がnew()メソッドに渡されるならば、 例外が発生します。

    my $obj = SomeClass->new(1); # Exception!

アクセッサにおいても同じです。

    $obj->foo(a => 1); # Execption!

メソッドのインポート

Object::Simpleのメソッドをインポートすることができます。 これは多重継承を行いたくない場合に便利です。

    package SomeClass;
    
    use Object::Simple qw/new attr/;
    
    __PACKAGE__->attr('foo');

new()はクラスにインポートされるのであって、 基底クラスから継承したわけではないので、 new()メソッドをオーバーライドすることはできない ということに注意してください。

メソッドチェーン

アクセサメソッドは、値を設定するために呼ばれたときに、 自分自分のオブジェクトを返却するので、メソッドチェーンを 行うことができます。

    $obj->foo(1)->bar(4)->baz(6);