-
Notifications
You must be signed in to change notification settings - Fork 0
Home
Botox – simple implementation of Modern Object Constructor with accessor, prototyping and default-filling of inheritanced values.
$VERSION 1.01
Botox предназначен для создания объектов с прототипируемыми управляемыми по доступности свойствами: write-protected или public И возможности установки этим свойствам дефолтных значений (как в прототипе, так и при создании объекта). Построение цепочки наследования производится на основе ::mro, методы и свойства переопределяются в потомке.
package Parent; use Botox qw(new); # yes, we are got constructor # default properties for ANY object of `Parent` class: # prop1_ro ISA 'write-protected' && prop2 ISA 'public' # and seting default value for each other sub _prototype_{ 'prop1_ro' => 1 , 'prop2' => 'abcde' }; # OR sub _prototype_{ return {'prop1_ro' => 1 , 'prop2' => 'abcde'} };
Botox – простой абстрактный конструктор, дающий возможность создавать объекты по прототипу и управлять их свойствами: write-protected или public. Кроме того он позволяет проставить свойствам значения по умолчанию (как в прототипе, так и при создании объекта).
Класс создается так:
package Parent;
use Botox qw(new); sub _prototype_{ 'prop1_ro' => 1 , 'prop2' => 'abcde' }; sub show_prop1{ # It`s poinlessly - indeed property IS A accessor itself my ( $self ) = @_; return $self->prop1; } sub set_prop1{ # It`s NEEDED for RO aka protected on write property my ( $self, $value ) = @_; $self->prop1($value); } sub parent_sub{ # It`s class method itself my $self = shift; return $self->prop1; } 1;
Экземпляр объекта создается так:
package Child; # change default value for prop1 my $foo = new Parent( { prop1 => 888888 } ); 1;
Собственно, это весь код для создания экземпляра.
use Data::Dumper; print Dumper($foo); Даст нам
$VAR1 = bless( { 'Parent::prop1' => 888888, 'Parent::prop2' => 'abcde' }, 'Parent' );
Свойства, описаные в конструкторе класса, могут наследоваться и иметь права доступа.
Право доступа указывается в имени свойства конструкцией bar + ‘_ro’ или ‘_rw’(default).
Право на доступ по чтению-записи является правилом по умолчанию, таким образом его указание не обязательно.
Напротив, ограничение прав “ro – только чтение” требует явного указания этого факта.
Далее для возможности работы с данным свойством НА ЗАПИСЬ из экземпляра объекта следует создать в классе акцессор, например:
eval{ $foo->prop1(-23) }; print $@."\n"; Даст нам что-то вроде:
Can`t change RO properties |prop1| to |-23| in object Parent from Child at ./test_more.t line 84
Для работы с данным свойством в родительском классе у нас был создан аксессор, который и следует использовать:
$foo->set_prop1(-23);
Создавая производный класс от родительского
package Child; use base 'Parent';
sub _prototype_{ return {'prop1' => 48, 'prop5' => 55 , 'prop8_ro' => 'tetete' }; }; 1;
В наследство мы получим ВСЕ методы Parent (что ожидаемо) и ВСЕ дефолтные свойства Parent (что неожиданно), и это правильная вещь.
Дефолтные значения И свойства доступа прототипа могут переопределяться в потомке, перезаписывая данные.
package GrandChild;
use Data::Dumper; my $baz = new Child(); print Dumper($baz); eval{$baz->prop8(-23)}; print $@."\n"; eval{$baz->prop1(1_000)}; print '$baz->prop1 = '.$baz->prop1()."\n";
Даст нам вот такой вывод:
$VAR1 = bless( { 'Child::prop5' => 55, 'Child::prop2' => 'abcde', 'Child::prop1' => 48, 'Child::prop8' => 'tetete' }, 'Child' );
Can`t change RO properties |prop8| to |-23| in object Child from GrandChild at ./test_more.t line 84 $baz->prop1 = 1000
То есть мы смогли получить новый класс Child на базе Parent со свойствами обоих классов, причем настройки верхнего уровня переопределяют родительские (включая права доступа).
Кроме того, возможна дефолтная инициализация свойств объекта:
package GrandChild;
use Data::Dumper; my $baz = new Child(prop1 => 99); # OR ({prop1 => 99}) as you wish print Dumper($baz);
Даст нам вот такой вывод:
$VAR1 = bless( { 'Child::prop5' => 55, 'Child::prop2' => 'abcde', 'Child::prop1' => 99, 'Child::prop8' => 'tetete' }, 'Child' );
При создании объекта возможно задавать значения как rw так и ro свойств(что логично).
Попытка задать значение несуществующего свойства даст ошибку.
Meettya <meettya@gmail.com>
Вероятно даже в таком объеме кода могут быть баги. Пожалуйста, сообщайте мне об их наличии по указаному e-mail или любым иным способом.
Moose, Mouse, Class::Accessor
Moscow, fall 2009.
This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. You may also can redistribute it and/or modify it under the terms of the Perl Artistic License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.