NAME
Class::Factory - Base class for dynamic factory classes
SYNOPSIS
package My::Factory;
use base qw( Class::Factory );
my %TYPES = ();
# SIMPLE: Let the parent know about our types
sub get_factory_map { return \%TYPES }
# FLEXIBLE: Let the parent know about our types
sub get_factory_type {
my ( $class, $type ) = @_;
return $TYPES{ $type };
}
sub set_factory_type {
my ( $class, $type, $factory_class ) = @_;
$TYPES{ $type } = $factory_class;
}
# LAZY LOADING: Allow subclasses to register a type without loading
# its class
my %REGISTERED = ();
# SIMPLE LAZY LOAD: Let the parent know about registered types
sub get_register_map { return \%REGISTERED }
# FLEXIBLE LAZY LOAD: Let the parent know about registered types
sub get_register_type {
my ( $class, $type ) = @_;
return $REGISTERED{ $type };
}
sub set_register_type {
my ( $class, $type, $factory_class ) = @_;
$REGISTERED{ $type } = $factory_class;
}
# Add our default types
# This type is loaded when the statement is run
My::Factory->add_factory_type( perl => 'My::Factory::Perl' );
# This type is loaded on the first request for type 'blech'
My::Factory->register_factory_type( blech => 'My::Factory::Blech' );
1;
# Adding a new factory type in code -- 'Other::Custom::Class' is
# brought in when 'add_factory_type()' is called
My::Factory->add_factory_type( custom => 'Other::Custom::Class' );
my $custom_object = My::Factory->new( 'custom', { this => 'that' } );
# Registering a new factory type in code --
# 'Other::Custom::ClassTwo' in brought in when 'new()' is called
# with type 'custom_two'
My::Factory->register_factory_type( custom_two => 'Other::Custom::ClassTwo' );
my $custom_object = My::Factory->new( 'custom_two', { this => 'that' } );
DESCRIPTION
This is a simple module that factory classes can use to generate new types of objects on the fly, providing a consistent interface to common groups of objects.
Factory classes are used when you have different implementations for the same set of tasks but may not know in advance what implementations you will be using. For instance, take configuration files. There are four basic operations you would want to do with any configuration file: read it in, lookup a value, set a value, write it out. There are also many different types of configuration files, and you may want users to be able to provide an implementation for their own home-grown configuration format.
With a factory class this is easy. To create the factory class, just subclass Class::Factory
and create an interface for your configuration serializer. Class::Factory
even provides a simple constructor for you:
package My::ConfigFactory;
use strict;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_map { return \%TYPES }
sub read { die "Define read() in implementation" }
sub write { die "Define write() in implementation" }
sub get { die "Define get() in implementation" }
sub set { die "Define set() in implementation" }
1;
And then users can add their own subclasses:
package My::CustomConfig;
use strict;
use base qw( My::ConfigFactory );
sub init {
my ( $self, $filename, $params ) = @_;
if ( $params->{base_url} ) {
$self->read_from_web( join( '/', $params->{base_url}, $filename ) );
}
else {
$self->read( $filename );
}
return $self;
}
sub read { ... implementation to read a file ... }
sub write { ... implementation to write a file ... }
sub get { ... implementation to get a value ... }
sub set { ... implementation to set a value ... }
sub read_from_web { ... implementation to read via http ... }
1;
(Normally you probably would not make your factory the same as your interface, but this is an abbreviated example.)
So now users can use the custom configuration with something like:
#!/usr/bin/perl
use strict;
use My::ConfigFactory;
My::ConfigFactory->add_factory_type( 'custom' => 'My::CustomConfig' );
my $config = My::ConfigFactory->new( 'custom', 'myconf.dat' );
This might not seem like a very big win, and for small standalone applications it is not. But when you develop large applications the add_factory_type()
step will almost certainly be done at application initialization time, hidden away from the eyes of the application developer. That developer will only know that she can access the different object types as if they are part of the system.
As you see in the example above, implementation for subclasses is very simple. The base class defines two methods for subclasses to use: get_factory_class()
and add_factory_type()
. Subclasses must define either get_factory_map()
or both get_factory_type()
and set_factory_type()
.
Registering Factory Types
As an additional feature, you can have your class accept registered types that get brought in only when requested. This lazy loading feature can be very useful when your factory offers many choices and users will only need one or two of them at a time, or when some classes the factory generates use libraries that some users may not have installed.
For example, say I have a factory that generates an object which parses GET/POST parameters. One type is straightforward CGI, one is Apache::Request. Many systems do not have Apache::Request installed so we do not want to 'use' the module whenever we create the factory.
Instead, we will register these types with the factory and only when that type is requested will be bring that implementation in. To extend our example above, we will use the configuration factory:
package My::ConfigFactory;
use strict;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_map { return \%TYPES }
my %REGISTERED = ();
sub get_register_map { return \%REGISTERED }
sub read { die "Define read() in implementation" }
sub write { die "Define write() in implementation" }
sub get { die "Define get() in implementation" }
sub set { die "Define set() in implementation" }
1;
We only added two lines. But now we can register configuration types for modules that the user might not have:
My::ConfigFactory->register_factory_type( SOAP => 'My::Config::SOAP' );
So the My::Config::SOAP
class will only get included at the first request for a configuration object of that type:
my $config = My::ConfigFactory->new( 'SOAP', 'http://myco.com/',
{ port => 8080, ... } );
METHODS
new( $type, @params )
This is a default constructor you can use. It is quite simple:
sub new {
my ( $pkg, $type, @params ) = @_;
my $class = $pkg->get_factory_class( $type );
my $self = bless( {}, $class );
return $self->init( @params );
}
We just create a new object of the class associated (from an earlier call to add_factory_type()
) with $type
and then call the init()
method of that object. The init()
method should return the object, or die on error.
get_factory_class( $object_type )
Usually called from a constructor when you want to lookup a class by a type and create a new object of $object_type
.
Returns: name of class. If a class matching $object_type
is not found, then a die()
is thrown.
add_factory_type( $object_type, $object_class )
Tells the factory to dynamically add a new type to its stable and brings in the class implementing that type using require()
. After running this the factory class will be able to create new objects of type $object_type
.
Returns: name of class added if successful. If the proper parameters are not given or if we cannot find $object_class
in @INC, then a die()
is thrown. A warn
ing is issued if the type has already been added.
register_factory_type( $object_type, $object_class )
Tells the factory to register a new factory type. This type will be dynamically included (using add_factory_type()
at the first request for an instance of that type.
Returns: name of class registered if successful. If the proper parameters are not given then a die()
is thrown. A warn
ing is issued if the type has already been registered.
SUBCLASSING
You can do this either the simple way or the flexible way. For most cases, the simple way will suffice:
package My::Factory;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_map { return \%TYPES }
If you want to also add the ability to register classes:
package My::Factory;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_map { return \%TYPES }
my %REGISTERED = ();
sub get_register_map { return \%REGISTERED }
If you elect to use the flexible way, you need to implement two methods:
package My::Factory;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_class {
my ( $class, $type ) = @_;
return $TYPES{ $type };
}
sub set_factory_class {
my ( $class, $type, $object_class ) = @_;
return $TYPES{ $type } = $object_class;
}
And if you want to use the flexible way along with registering types:
package My::Factory;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_class {
my ( $class, $type ) = @_;
return $TYPES{ $type };
}
sub set_factory_class {
my ( $class, $type, $object_class ) = @_;
return $TYPES{ $type } = $object_class;
}
my %REGISTERED = ();
sub get_register_class {
my ( $class, $type ) = @_;
return $REGISTERED{ $type };
}
sub set_register_class {
my ( $class, $type, $object_class ) = @_;
return $REGISTERED{ $type } = $object_class;
}
How these methods work is entirely up to you -- maybe get_factory_class()
does a lookup in some external resource before returning the class. Whatever floats your boat.
USAGE
Common Pattern
This is a very common pattern. Subclasses create an init()
method that gets called when the object is created:
package My::Factory;
use strict;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_map { return \%TYPES }
my %REGISTERED = ();
sub get_register_map { return \%REGISTERED }
1;
And here is what a subclass might look like:
package My::Subclass;
use strict;
use base qw( Class::Accessor );
my @CONFIG_FIELDS = qw( status created_on created_by updated_on updated_by );
my @FIELDS = ( 'filename', @CONFIG_FIELDS );
My::Subclass->mk_accessors( @FIELDS );
# Note: we've taken the flattened C<@params> passed in and assigned
# the first element as C<$filename> and the other element as a
# hashref C<$params>
sub init {
my ( $self, $filename, $params ) = @_;
unless ( -f $filename ) {
die "Filename [$filename] does not exist. Object cannot be created";
}
$self->filename( filename );
$self->read_file_contents;
foreach my $field ( @CONFIG_FIELDS ) {
$self->{ $field } = $params->{ $field } if ( $params->{ $field } );
}
return $self;
}
The parent class (My::Factory
) has made as part of its definition that the only parameters to be passed to the init()
method are $filename
and $params
, in that order. It could just as easily have specified a single hashref parameter.
These sorts of specifications are informal and not enforced by this Class::Factory
.
Registering Types
Many times you will want the parent class to automatically register the types it knows about:
package My::Factory;
use strict;
use base qw( Class::Factory );
my %TYPES = ();
sub get_factory_map { return \%TYPES }
my %REGISTERED = ();
sub get_register_map { return \%REGISTERED }
My::Factory->register_factory_type( type1 => 'My::Impl::Type1' );
My::Factory->register_factory_type( type2 => 'My::Impl::Type1' );
1;
This allows the default types to be registered when the factory is initialized. So you can use the default implementations without any more registering/adding:
#!/usr/bin/perl
use strict;
use My::Factory;
my $impl1 = My::Factory->new( 'type1' );
my $impl2 = My::Factory->new( 'type2' );
COPYRIGHT
Copyright (c) 2002 Chris Winters. All rights reserved.
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
SEE ALSO
"Design Patterns", by Erich Gamma, Richard Helm, Ralph Johnson and John Vlissides. Addison Wesley Longman, 1995. Specifically, the 'Factory Method' pattern, pp. 107-116.
AUTHOR
Chris Winters <chris@cwinters.com>