NAME
DCI - Collection of utilities for writing perl code that fits the DCI methodology.
DESCRIPTION
The DCI concept was created by Trygve Reenskaug, (inventor of MVC) and James Coplien.
DCI Stands for Data, Context, Interactions. It was created to solve the problem of unpredictable emergent behavior in networks of interacting objects. This problem shows itself in complex OOP projects, most commonly in projects with deep polymorphism. This is a problem that Procedural/Imperative Programming does not have.
DCI does not replace OOP, instead it augments it with lessons learned from looking back at Procedural Programming. It defines a way to encapsulate use cases into a single place. This provides an advantage to the programmer by reducing the number of interactions that need to be tracked. Another advantage is the reduction of side-effects between contexts.
Another way to look at it is that a DCI implementation is much more maintainable as a project matures. Changes to requirements and additional features cause clean OOP project to degrade into spaghetti. DCI on the other hand maintains code clarity under changing requirements.
MORE ABOUT DCI
Look at DCI::Introduction for a complete introduction into DCI concepts.
SYNOPSIS
This synopsis is very basic, see individual modules for better module-specific examples. Also see DCI::Introduction if you do not yet understand DCI concepts.
DUMB DATA CLASS
package Test::Data;
use strict;
use warnings;
sub new {
my $class = shift;
my ( $value ) = @_;
return bless( \$value, $class );
}
sub get_text { ${ $_[0] } }
sub set_text {
my $self = shift;
my ( $new_val ) = @_;
$$self = $new_val;
}
1;
TRIVIAL CAST CLASS (ROLE)
package Test::Cast;
use strict;
use warnings;
use DCI qw/Cast/;
# Delegate the text() method to the 'core'
delegate qw/text/;
# Restrict core to being a Test::Data object.
require_core qw/Test::Data/;
# Generate some accessors which keep state in the Cast, but do not interfer
# with the Core object.
accessors qw/cache/;
# Add a method to our cast
sub render {
my $self = shift;
my $cached = $self->cache;
# text() is delgated to the underlying 'core'
$cached = $self->cache(
"The text is: '" . $self->text() . "'\n"
) unless $cached;
return $cached;
}
1;
A CONTEXT
package Test::Context;
use strict;
use warnings;
use DCI qw/Context/;
use Test::Cast;
# Declare some required member objects
cast first => 'Test::Cast',
last => 'Test::Cast';
# export a function named 'quick_render' that constructs the object from
# arguments, and returns the result of running the 'render()' method on the
# created object.
sugar quick_render => (
method => 'render',
ordered => [qw/ first last /]
);
# A sugar export that takes named arguments.
sugar parametric_render => 'render';
# Method we use to kick-off the encapsulated task of the Context.
sub render { shift->first->render }
1;
PUTTING IT ALL TOGETHER
use Test::More;
use Test::Context qw/quick_render/;
is(
quick_render( "String A", "String B" ),
"The text is: 'String A'\nThe text is: 'String B'",
"Rendered first and last strings using quick sugar"
);
is(
parametric_render(
first => "String A",
last => "String B",
),
"The text is: 'String A'\nThe text is: 'String B'",
"Rendered first and last strings using named parameters"
);
my $it = Test::Context->new(
first => "String A",
last => "String B",
);
is(
$it->render,
"The text is: 'String A'\nThe text is: 'String B'",
"Rendered first and last strings from onbject instance"
);
1;
OBJECT TYPES
CONTEXT
SUGAR FUNCTIONS
See DCI::Meta::Context which actually exports the sugar.
METHODS
See DCI::Context the base class for all Casts.
CAST
SUGAR FUNCTIONS
See DCI::Meta::Cast which actually exports the sugar.
METHODS
See DCI::Cast the base class for all Casts.
API STABILITY
Versions below 0.011 were a prototype, the API was subject to flux. For 0.011 the API has been completely re-written using the lessons learned from the old API.
As of 0.011, the API can be considered stable. This means that changes which break backwards compatibility will be few and far between.
SEE ALSO
- DCI::Meta::Context
-
Meta class for contexts
- DCI::Meta::Cast
-
Meta class for casts
- DCI::Context
-
Base class for contexts
- DCI::Cast
-
Base class for casts
ACHNOWLEDGEMENTS
The DCI concept was created by Trygve Reenskaug, (inventor of MVC) and James Coplien.
AUTHORS
Chad Granum exodist7@gmail.com
COPYRIGHT
Copyright (C) 2011 Chad Granum
DCI is free software; Standard perl licence.
DCI 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 license for more details.