NAME DBIx::FlexibleBinding - Greater flexibility on statement placeholder choice and data binding VERSION version 2.0.1 SYNOPSIS This module extends the DBI allowing you choose from a variety of supported parameter placeholder and binding patterns as well as offering simplified ways to interact with datasources, while improving general readability. ######################################################### # SCENARIO 1 # # A connect followed by a prepare-execute-process cycle # ######################################################### use DBIx::FlexibleBinding; use constant DSN => 'dbi:mysql:test;host=127.0.0.1'; use constant SQL => << '//'; SELECT solarSystemName AS name FROM mapsolarsystems WHERE regional = :is_regional AND security >= :minimum_security // # Pretty standard connect, just with the new DBI subclass ... # my $dbh = DBIx::FlexibleBinding->connect(DSN, '', '', { RaiseError => 1 }); # Prepare statement using named placeholders (not bad for MySQL, eh) ... # my $sth = $dbh->prepare(SQL); # Execute the statement (parameter binding is automatic) ... # my $rv = $sth->execute(is_regional => 1, minimum_security => 1.0); # Fetch and transform rows with a blocking callback to get only the data you # want without cluttering the place up with intermediate state ... # my @system_names = $sth->getrows_hashref(callback { $_->{name} }); ############################################################################ # SCENARIO 2 # # Let's simplify the previous scenario using the database handle's version # # of that getrows_hashref method. # ############################################################################ use DBIx::FlexibleBinding -alias => 'DFB'; use constant DSN => 'dbi:mysql:test;host=127.0.0.1'; use constant SQL => << '//'; SELECT solarSystemName AS name FROM mapsolarsystems WHERE regional = :is_regional AND security >= :minimum_security // # Pretty standard connect, this time with the DBI subclass package alias ... # my $dbh = DFB->connect(DSN, '', '', { RaiseError => 1 }); # Cut out the middle men ... # my @system_names = $dbh->getrows_hashref(SQL, is_regional => 1, minimum_security => 1.0, callback { $_->{name} }); ############################################################################# # SCENARIO 3 # # The subclass import method provides a versatile mechanism for simplifying # # matters further. # ############################################################################# use DBIx::FlexibleBinding -subs => [ 'MyDB' ]; use constant DSN => 'dbi:mysql:test;host=127.0.0.1'; use constant SQL => << '//'; SELECT solarSystemName AS name FROM mapsolarsystems WHERE regional = :is_regional AND security >= :minimum_security // # MyDB will represent our datasource; initialise it ... # MyDB DSN, '', '', { RaiseError => 1 }; # Cut out the middle men and some of the line-noise, too ... # my @system_names = MyDB(SQL, is_regional => 1, minimum_security => 1.0, callback { $_->{name} }); DESCRIPTION This module subclasses the DBI to provide improvements and greater flexibility in the following areas: * Parameter placeholders and data binding * Data retrieval and processing * Accessing and interacting with datasources Parameter placeholders and data binding This module provides support for a wider range of parameter placeholder and data-binding schemes. As well as continued support for the simple positional placeholders ("?"), additional support is provided for numeric placeholders (":N" and "?N"), and named placeholders (":NAME" and @NAME). As for the process of binding data values to parameters: that is, by default, now completely automated, removing a significant part of the workload from the prepare-bind-execute cycle. It is, however, possible to swtch off automatic data-binding globally and on a statement-by-statement basis. The following familiar operations have been modified to accommodate all of these changes, though developers continue to use them as they always have done: * "$DATABASE_HANDLE->prepare($STATEMENT, \%ATTR);" * "$DATABASE_HANDLE->do($STATEMENT, \%ATTR, @DATA);" * "$STATEMENT_HANDLE->bind_param($NAME_OR_POSITION, $VALUE, \%ATTR);" * "$STATEMENT_HANDLE->execute(@DATA);" Data retrieval and processing Four new methods, each available for database and statement handles, have been implemented: * "getrow_arrayref" * "getrow_hashref" * "getrows_arrayref" * "getrows_hashref" These methods complement DBI's existing fetch methods, providing new ways to retrieve and process data. Accessing and interacting with datasources The module's "-subs" import option may be used to create subroutines, during the compile phase, and export them to the caller's namespace for use later as representations of database and statement handles. * Use for connecting to datasources use DBIx::FlexibleBinding -subs => [ 'MyDB' ]; # Pass in any set of well-formed DBI->connect(...) arguments to associate # your name with a live database connection ... # MyDB( 'dbi:mysql:test;host=127.0.0.1', '', '', { RaiseError => 1 } ); # Or, simply pass an existing database handle as the only argument ... # MyDB($dbh); * Use them to represent database handles use DBIx::FlexibleBinding -subs => [ 'MyDB' ]; use constant SQL => << '//'; SELECT * FROM mapsolarsystems WHERE regional = :is_regional AND security >= :minimum_security // MyDB( 'dbi:mysql:test;host=127.0.0.1', '', '', { RaiseError => 1 } ); # If your name is already associated with a database handle then just call # it with no parameters to use it as such ... # my $sth = MyDB->prepare(SQL); * Use them to represent statement handles use DBIx::FlexibleBinding -subs => [ 'MyDB', 'solar_systems' ]; use constant SQL => << '//'; SELECT * FROM mapsolarsystems WHERE regional = :is_regional AND security >= :minimum_security // MyDB( 'dbi:mysql:test;host=127.0.0.1', '', '', { RaiseError => 1 } ); my $sth = MyDB->prepare(SQL); # Simply call the statement handle proxy, passing a statement handle in as # the only argument ... # solar_systems($sth); * Use to interact with the represented database and statement handles use DBIx::FlexibleBinding -subs => [ 'MyDB', 'solar_systems' ]; use constant SQL => << '//'; SELECT * FROM mapsolarsystems WHERE regional = :is_regional AND security >= :minimum_security // MyDB( 'dbi:mysql:test;host=127.0.0.1', '', '', { RaiseError => 1 } ); # Use the database handle proxy to prepare, bind and execute statements, then # retrieve the results ... # # Use the database handle proxy to prepare, bind and execute statements, then # retrieve the results ... # my $array_of_hashrefs = MyDB(SQL, is_regional => 1, minimum_security => 1.0); # In list context, results come back as lists ... # my @array_of_hashrefs = MyDB(SQL, is_regional => 1, minimum_security => 1.0); # Using -subs also relaxes strict 'subs' in the caller's scope, so pretty-up # void context calls by losing the parentheses, if you wish to use callbacks # to process the results ... # MyDB SQL, is_regional => 1, minimum_security => 1.0, callback { printf "%-16s %.1f\n", $_->{solarSystemName}, $_->{security}; }; # You can use proxies to represent statements, too. Simply pass in a statement # handle as the only argument ... # my $sth = MyDB->prepare(SQL); solar_systems($sth); # Using "solar_systems" as statement proxy. # Now, when called with other types of arguments, those argument values are # bound and the statement is executed ... # my $array_of_hashrefs = solar_systems(is_regional => 1, minimum_security => 1.0); # In list context, results come back as lists ... # my @array_of_hashrefs = solar_systems(is_regional => 1, minimum_security => 1.0); # Statements requiring no parameters cannot be used in this manner because # making a call to a statement proxy with an arity of zero results in the # statement handle being returned. In this situation, use something like # undef as an argument (it will be ignored in this particular instance) ... # my $rv = statement_proxy(undef); # # Meh, you can't win 'em all! PACKAGE GLOBALS $DBIx::FlexibleBinding::AUTO_BINDING_ENABLED A boolean setting used to determine whether or not automatic binding is enabled or disabled globally. The default setting is "1" (*enabled*). IMPORT TAGS AND OPTIONS -alias This option may be used by the caller to select an alias to use for this package's unwieldly namespace. use DBIx::FlexibleBinding -alias => 'DBIF'; my $dbh = DBIF->connect('dbi:SQLite:test.db', '', ''); -subs This option may be used to create subroutines, during the compile phase, in the caller's namespace to be used as representations of database and statement handles. use DBIx::FlexibleBinding -subs => [ 'MyDB' ]; # Initialise by passing in a valid set of DBI->connect(...) arguments. # The database handle will be the return value. # MyDB 'dbi:mysql:test;host=127.0.0.1', '', '', { RaiseError => 1 }; # Or, initialise by passing in a DBI database handle. # The handle is also the return value. # MyDB $dbh; # Once initialised, use the subroutine as you would a DBI database handle. # my $statement = << '//'; SELECT solarSystemName AS name FROM mapsolarsystems WHERE security >= :minimum_security // my $sth = MyDB->prepare($statement); # Or use it as an expressive time-saver! # my $array_of_hashrefs = MyDB($statement, security => 1.0); my @system_names = MyDB($statement, minimum_security => 1.0, callback { return $_->{name}; }); MyDB $statement, minimum_security => 1.0, callback { my ($row) = @_; print "$row->{name}\n"; }; Use of this option automatically relaxes "strict 'subs'" for the remainder of scope containing the "use" directive. That is unless "use strict 'subs'" or "use strict" appears afterwards. CLASS METHODS connect $dbh = DBIx::FlexibleBinding->connect($data_source, $user, $pass) or die $DBI::errstr; $dbh = DBIx::FlexibleBinding->connect($data_source, $user, $pass, \%attr) or die $DBI::errstr; Establishes a database connection, or session, to the requested data_source and returns a database handle object if the connection succeeds or undef if it does not. Refer to for a more detailed description of this method. DATABASE HANDLE METHODS do $rows = $dbh->do($statement_string) or die $dbh->errstr; $rows = $dbh->do($statement_string, @bind_values) or die $dbh->errstr; $rows = $dbh->do($statement_string, \%attr) or die $dbh->errstr; $rows = $dbh->do($statement_string, \%attr, @bind_values) or die $dbh->errstr; $rows = $dbh->do($statement_handle) or die $dbh->errstr; $rows = $dbh->do($statement_handle, @bind_values) or die $dbh->errstr; Prepares (if necessary) and executes a single statement. Returns the number of rows affected or undef on error. A return value of -1 means the number of rows is not known, not applicable, or not available. When no rows have been affected this method continues the "DBI" tradition of returning 0E0 on successful execution and "undef" on failure. The "do" method accepts optional callbacks for further processing of the result. The "do" implementation provided by this module allows for some minor deviations in usage over the standard "DBI" implementation. In spite of this, the new method may be used just like the original. Refer to for a more detailed description of this method. Examples 1. Statement attributes are now optional: $sql = << '//'; UPDATE employees SET salary = :salary WHERE employee_id = :employee_id // $dbh->do($sql, employee_id => 52, salary => 35_000) or die $dbh->errstr; A reference to the statement attributes hash is no longer required, even if it's empty. If, however, a hash reference is supplied as the first parameter then it would be used for that purpose. 2. Prepared statements now may be re-used: $sth = $dbh->prepare(<< '//'); UPDATE employees SET salary = ? WHERE employee_id = ? // $dbh->do($sth, 35_000, 52) or die $dbh->errstr; A prepared statement may also be used in lieu of a statement string. In such cases, referencing a statement attributes hash is neither required nor expected. prepare $sth = $dbh->prepare($statement_string); $sth = $dbh->prepare($statement_string, \%attr); Prepares a statement for later execution by the database engine and returns a reference to a statement handle object. Refer to for a more detailed description of this method. Examples 1. Prepare a statement using positional placeholders: $sql = << '//'; UPDATE employees SET salary = ? WHERE employee_id = ? // $sth = $dbh->prepare($sql); 2. Prepare a statement using named placeholders: *(Yes, even for those MySQL connections!)* $sql = << '//'; UPDATE employees SET salary = :salary WHERE employee_id = :employee_id // $sth = $dbh->prepare($sql); getrows_arrayref *(database handles)* $results = $dbh->getrows_arrayref($statement_string, @bind_values); @results = $dbh->getrows_arrayref($statement_string, @bind_values); $results = $dbh->getrows_arrayref($statement_string, \%attr, @bind_values); @results = $dbh->getrows_arrayref($statement_string, \%attr, @bind_values); $results = $dbh->getrows_arrayref($statement_handle, @bind_values); @results = $dbh->getrows_arrayref($statement_handle, @bind_values); Prepares (if necessary) and executes a single statement with the specified data bindings and fetches the result set as an array of array references. The "getrows_arrayref" method accepts optional callbacks for further processing of the results by the caller. Examples 1. Prepare, execute it then get the results as a reference: $sql = << '//'; SELECT solarSystemName AS name , security FROM mapsolarsystems WHERE regional = 1 AND security >= :minimum_security // $systems = $dbh->getrows_arrayref($sql, minimum_security => 1.0); # Returns a structure something like this: # # [ [ 'Kisogo', '1' ], # [ 'New Caldari', '1' ], # [ 'Amarr', '1' ], # [ 'Bourynes', '1' ], # [ 'Ryddinjorn', '1' ], # [ 'Luminaire', '1' ], # [ 'Duripant', '1' ], # [ 'Yulai', '1' ] ] 2. Re-use a prepared statement, execute it then return the results as a list: We'll use the query from Example 1 but have the results returned as a list for further processing by the caller. $sth = $dbh->prepare($sql); @systems = $dbh->getrows_arrayref($sql, minimum_security => 1.0); for my $system (@systems) { printf "%-11s %.1f\n", @$system; } # Output: # # Kisogo 1.0 # New Caldari 1.0 # Amarr 1.0 # Bourynes 1.0 # Ryddinjorn 1.0 # Luminaire 1.0 # Duripant 1.0 # Yulai 1.0 3. Re-use a prepared statement, execute it then return modified results as a reference: We'll use the query from Example 1 but have the results returned as a list for further processing by a caller who will be using callbacks to modify those results. $sth = $dbh->prepare($sql); $systems = $dbh->getrows_arrayref($sql, minimum_security => 1.0, callback { my ($row) = @_; return sprintf("%-11s %.1f\n", @$row); }); # Returns a structure something like this: # # [ 'Kisogo 1.0', # 'New Caldari 1.0', # 'Amarr 1.0', # 'Bourynes 1.0', # 'Ryddinjorn 1.0', # 'Luminaire 1.0', # 'Duripant 1.0', # 'Yulai 1.0' ] getrows_hashref *(database handles)* $results = $dbh->getrows_hashref($statement_string, @bind_values); @results = $dbh->getrows_hashref($statement_string, @bind_values); $results = $dbh->getrows_hashref($statement_string, \%attr, @bind_values); @results = $dbh->getrows_hashref($statement_string, \%attr, @bind_values); $results = $dbh->getrows_hashref($statement_handle, @bind_values); @results = $dbh->getrows_hashref($statement_handle, @bind_values); Prepares (if necessary) and executes a single statement with the specified data bindings and fetches the result set as an array of hash references. The "getrows_hashref" method accepts optional callbacks for further processing of the results by the caller. Examples 1. Prepare, execute it then get the results as a reference: $sql = << '//'; SELECT solarSystemName AS name , security FROM mapsolarsystems WHERE regional = 1 AND security >= :minimum_security // $systems = $dbh->getrows_hashref($sql, minimum_security => 1.0); # Returns a structure something like this: # # [ { name => 'Kisogo', security => '1' }, # { name => 'New Caldari', security => '1' }, # { name => 'Amarr', security => '1' }, # { name => 'Bourynes', security => '1' }, # { name => 'Ryddinjorn', security => '1' }, # { name => 'Luminaire', security => '1' }, # { name => 'Duripant', security => '1' }, # { name => 'Yulai', security => '1' } ] 2. Re-use a prepared statement, execute it then return the results as a list: We'll use the query from Example 1 but have the results returned as a list for further processing by the caller. $sth = $dbh->prepare($sql); @systems = $dbh->getrows_hashref($sql, minimum_security => 1.0); for my $system (@systems) { printf "%-11s %.1f\n", @{$system}{'name', 'security'}; # Hash slice } # Output: # # Kisogo 1.0 # New Caldari 1.0 # Amarr 1.0 # Bourynes 1.0 # Ryddinjorn 1.0 # Luminaire 1.0 # Duripant 1.0 # Yulai 1.0 3. Re-use a prepared statement, execute it then return modified results as a reference: We'll use the query from Example 1 but have the results returned as a list for further processing by a caller who will be using callbacks to modify those results. $sth = $dbh->prepare($sql); $systems = $dbh->getrows_hashref($sql, minimum_security => 1.0, callback { sprintf("%-11s %.1f\n", @{$_}{'name', 'security'}); # Hash slice }); # Returns a structure something like this: # # [ 'Kisogo 1.0', # 'New Caldari 1.0', # 'Amarr 1.0', # 'Bourynes 1.0', # 'Ryddinjorn 1.0', # 'Luminaire 1.0', # 'Duripant 1.0', # 'Yulai 1.0' ] getrows *(database handles)* $results = $dbh->getrows($statement_string, @bind_values); @results = $dbh->getrows($statement_string, @bind_values); $results = $dbh->getrows($statement_string, \%attr, @bind_values); @results = $dbh->getrows($statement_string, \%attr, @bind_values); $results = $dbh->getrows($statement_handle, @bind_values); @results = $dbh->getrows$statement_handle, @bind_values); Alias for "getrows_hashref". If array references are preferred, have the symbol table glob point alias the "getrows_arrayref" method. The "getrows" method accepts optional callbacks for further processing of the results by the caller. getrow_arrayref *(database handles)* $result = $dbh->getrow_arrayref($statement_string, @bind_values); $result = $dbh->getrow_arrayref($statement_string, \%attr, @bind_values); $result = $dbh->getrow_arrayref($statement_handle, @bind_values); Prepares (if necessary) and executes a single statement with the specified data bindings and fetches the first row as an array reference. The "getrow_arrayref" method accepts optional callbacks for further processing of the result by the caller. getrow_hashref *(database handles)* $result = $dbh->getrow_hashref($statement_string, @bind_values); $result = $dbh->getrow_hashref($statement_string, \%attr, @bind_values); $result = $dbh->getrow_hashref($statement_handle, @bind_values); Prepares (if necessary) and executes a single statement with the specified data bindings and fetches the first row as a hash reference. The "getrow_hashref" method accepts optional callbacks for further processing of the result by the caller. getrow *(database handles)* $result = $dbh->getrow($statement_string, @bind_values); $result = $dbh->getrow($statement_string, \%attr, @bind_values); $result = $dbh->getrow($statement_handle, @bind_values); Alias for "getrow_hashref". If array references are preferred, have the symbol table glob point alias the "getrows_arrayref" method. The "getrow" method accepts optional callbacks for further processing of the result by the caller. STATEMENT HANDLE METHODS bind_param $sth->bind_param($param_num, $bind_value) $sth->bind_param($param_num, $bind_value, \%attr) $sth->bind_param($param_num, $bind_value, $bind_type) $sth->bind_param($param_name, $bind_value) $sth->bind_param($param_name, $bind_value, \%attr) $sth->bind_param($param_name, $bind_value, $bind_type) The "bind_param" method associates (binds) a value to a placeholder embedded in the prepared statement. The implementation provided by this module allows the use of parameter names, if appropriate, in addition to parameter positions. *Refer to for a more detailed explanation of how to use this method*. execute $rv = $sth->execute() or die $DBI::errstr; $rv = $sth->execute(@bind_values) or die $DBI::errstr; Perform whatever processing is necessary to execute the prepared statement. An "undef" is returned if an error occurs. A successful call returns true regardless of the number of rows affected, even if it's zero. Refer to for a more detailed description of this method. Examples Use prepare, execute and getrow_hashref with a callback to modify my data: use strict; use warnings; use DBIx::FlexibleBinding -subs => [ 'TestDB' ]; use Data::Dumper; use Test::More; $Data::Dumper::Terse = 1; $Data::Dumper::Indent = 1; TestDB 'dbi:mysql:test', '', '', { RaiseError => 1 }; my $sth = TestDB->prepare(<< '//'); SELECT solarSystemID AS id , solarSystemName AS name , security FROM mapsolarsystems WHERE solarSystemName RLIKE "^U[^0-9\-]+$" ORDER BY id, name, security DESC LIMIT 5 // $sth->execute() or die $DBI::errstr; my @rows; my @callback_list = ( callback { my ($row) = @_; $row->{filled_with} = ( $row->{security} >= 0.5 ) ? 'Carebears' : 'Yarrbears'; $row->{security} = sprintf('%.1f', $row->{security}); return $row; } ); while ( my $row = $sth->getrow_hashref(@callback_list) ) { push @rows, $row; } my $expected_result = [ { 'name' => 'Uplingur', 'filled_with' => 'Yarrbears', 'id' => '30000037', 'security' => '0.4' }, { 'security' => '0.4', 'id' => '30000040', 'name' => 'Uzistoon', 'filled_with' => 'Yarrbears' }, { 'name' => 'Usroh', 'filled_with' => 'Carebears', 'id' => '30000068', 'security' => '0.6' }, { 'filled_with' => 'Yarrbears', 'name' => 'Uhtafal', 'id' => '30000101', 'security' => '0.5' }, { 'security' => '0.3', 'id' => '30000114', 'name' => 'Ubtes', 'filled_with' => 'Yarrbears' } ]; is_deeply( \@rows, $expected_result, 'iterate' ) and diag( Dumper(\@rows) ); done_testing(); iterate $iterator = $sth->iterate() or die $DBI::errstr; $iterator = $sth->iterate(@bind_values) or die $DBI::errstr; Perform whatever processing is necessary to execute the prepared statement. An "undef" is returned if an error occurs. A successful call returns an iterator which can be used to traverse the result set. Examples 1. Using an iterator and callbacks to process the result set: use strict; use warnings; use DBIx::FlexibleBinding -subs => [ 'TestDB' ]; use Data::Dumper; use Test::More; $Data::Dumper::Terse = 1; $Data::Dumper::Indent = 1; my @drivers = grep { /^SQLite$/ } DBI->available_drivers(); SKIP: { skip("iterate tests (No DBD::SQLite installed)", 1) unless @drivers; TestDB "dbi:SQLite:test.db", '', '', { RaiseError => 1 }; my $sth = TestDB->prepare(<< '//'); SELECT solarSystemID AS id , solarSystemName AS name , security FROM mapsolarsystems WHERE solarSystemName REGEXP "^U[^0-9\-]+$" ORDER BY id, name, security DESC LIMIT 5 // # Iterate over the result set # --------------------------- # We also queue up a sneaky callback to modify each row of data as it # is fetched from the result set. my $it = $sth->iterate( callback { my ($row) = @_; $row->{filled_with} = ( $row->{security} >= 0.5 ) ? 'Carebears' : 'Yarrbears'; $row->{security} = sprintf('%.1f', $row->{security}); return $row; } ); my @rows; while ( my $row = $it->() ) { push @rows, $row; } # Done, now check the results ... my $expected_result = [ { 'name' => 'Uplingur', 'filled_with' => 'Yarrbears', 'id' => '30000037', 'security' => '0.4' }, { 'security' => '0.4', 'id' => '30000040', 'name' => 'Uzistoon', 'filled_with' => 'Yarrbears' }, { 'name' => 'Usroh', 'filled_with' => 'Carebears', 'id' => '30000068', 'security' => '0.6' }, { 'filled_with' => 'Yarrbears', 'name' => 'Uhtafal', 'id' => '30000101', 'security' => '0.5' }, { 'security' => '0.3', 'id' => '30000114', 'name' => 'Ubtes', 'filled_with' => 'Yarrbears' } ]; is_deeply( \@rows, $expected_result, 'iterate' ) and diag( Dumper(\@rows) ); } done_testing(); In this example, we're traversing the result set using an iterator. As we iterate through the result set, a callback is applied to each row and we're left with an array of transformed rows. 2. Using an iterator's "for_each" method and callbacks to process the result set: use strict; use warnings; use DBIx::FlexibleBinding -subs => [ 'TestDB' ]; use Data::Dumper; use Test::More; $Data::Dumper::Terse = 1; $Data::Dumper::Indent = 1; my @drivers = grep { /^SQLite$/ } DBI->available_drivers(); SKIP: { skip("iterate tests (No DBD::SQLite installed)", 1) unless @drivers; TestDB "dbi:SQLite:test.db", '', '', { RaiseError => 1 }; my $sth = TestDB->prepare(<< '//'); SELECT solarSystemID AS id , solarSystemName AS name , security FROM mapsolarsystems WHERE solarSystemName REGEXP "^U[^0-9\-]+$" ORDER BY id, name, security DESC LIMIT 5 // # Iterate over the result set # --------------------------- # This time around we call the iterator's "for_each" method to process # the data. Bonus: we haven't had to store the iterator anywhere or # pre-declare an empty array to accommodate our rows. my @rows = $sth->iterate->for_each( callback { my ($row) = @_; $row->{filled_with} = ( $row->{security} >= 0.5 ) ? 'Carebears' : 'Yarrbears'; $row->{security} = sprintf('%.1f', $row->{security}); return $row; } ); # Done, now check the results ... my $expected_result = [ { 'name' => 'Uplingur', 'filled_with' => 'Yarrbears', 'id' => '30000037', 'security' => '0.4' }, { 'security' => '0.4', 'id' => '30000040', 'name' => 'Uzistoon', 'filled_with' => 'Yarrbears' }, { 'name' => 'Usroh', 'filled_with' => 'Carebears', 'id' => '30000068', 'security' => '0.6' }, { 'filled_with' => 'Yarrbears', 'name' => 'Uhtafal', 'id' => '30000101', 'security' => '0.5' }, { 'security' => '0.3', 'id' => '30000114', 'name' => 'Ubtes', 'filled_with' => 'Yarrbears' } ]; is_deeply( \@rows, $expected_result, 'iterate' ) and diag( Dumper(\@rows) ); } done_testing(); Like the previous example, we're traversing the result set using an iterator but this time around we have done away with $it in favour of calling the iterator's own "for_each" method. The callback we were using to process each row of the result set has now been passed into the "for_each" method also eliminating a "while" loop and an empty declaration for @rows. getrows_arrayref *(database handles)* $results = $sth->getrows_arrayref(); @results = $sth->getrows_arrayref(); Fetches the entire result set as an array of array references. The "getrows_arrayref" method accepts optional callbacks for further processing of the results by the caller. getrows_hashref *(database handles)* $results = $sth->getrows_hashref(); @results = $sth->getrows_hashref(); Fetches the entire result set as an array of hash references. The "getrows_hashref" method accepts optional callbacks for further processing of the results by the caller. getrows *(database handles)* $results = $sth->getrows(); @results = $sth->getrows(); Alias for "getrows_hashref". If array references are preferred, have the symbol table glob point alias the "getrows_arrayref" method. The "getrows" method accepts optional callbacks for further processing of the results by the caller. getrow_arrayref *(database handles)* $result = $sth->getrow_arrayref(); Fetches the next row as an array reference. Returns "undef" if there are no more rows available. The "getrow_arrayref" method accepts optional callbacks for further processing of the result by the caller. getrow_hashref *(database handles)* $result = $sth->getrow_hashref(); Fetches the next row as a hash reference. Returns "undef" if there are no more rows available. The "getrow_hashref" method accepts optional callbacks for further processing of the result by the caller. getrow *(database handles)* $result = $sth->getrow(); Alias for "getrow_hashref". If array references are preferred, have the symbol table glob point alias the "getrows_arrayref" method. The "getrow" method accepts optional callbacks for further processing of the result by the caller. EXPORTS The following symbols are exported by default: callback To enable the namespace using this module to take advantage of the callbacks, which are one of its main features, without the unnecessary burden of also including the module that provides the feature *(see Params::Callbacks for more detailed information)*. SEE ALSO * DBI * Params::Callbacks REPOSITORY * * BUGS Please report any bugs or feature requests to "bug-dbix-anybinding at rt.cpan.org", or through the web interface at . I will be notified, and then you'll automatically be notified of progress on your bug as I make changes. SUPPORT You can find documentation for this module with the perldoc command. perldoc DBIx::FlexibleBinding You can also look for information at: * RT: CPAN's request tracker (report bugs here) * AnnoCPAN: Annotated CPAN documentation * CPAN Ratings * Search CPAN ACKNOWLEDGEMENTS Many, many thanks to the CPANTesters network. Test data set extracted from Fuzzwork's MySQL conversion of CCP's EVE Online Static Data Export: * Fuzzwork * EVE Online Eternal gratitude to GitHub contributors: * Syohei Yoshida AUTHOR Iain Campbell COPYRIGHT AND LICENSE This software is copyright (c) 2012-2015 by Iain Campbell. This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.