NAME
Hash::Subset - Produce subset of a hash
VERSION
This document describes version 0.007 of Hash::Subset (from Perl distribution Hash-Subset), released on 2022-07-27.
SYNOPSIS
use Hash::Subset qw( hash_subset hashref_subset hash_subset_without hashref_subset_without merge_hash_subset merge_overwrite_hash_subset merge_ignore_hash_subset merge_hash_subset_without merge_overwrite_hash_subset_without merge_ignore_hash_subset_without);# using keys specified in an arraymy %subset = hash_subset ({a=>1, b=>2, c=>3}, ['b','c','d']); # => (b=>2, c=>3)my $subset = hashref_subset({a=>1, b=>2, c=>3}, ['b','c','d']); # => {b=>2, c=>3}# using keys specified in another hashmy %subset = hash_subset ({a=>1, b=>2, c=>3}, {b=>20, c=>30, d=>40}); # => (b=>2, c=>3)my $subset = hashref_subset({a=>1, b=>2, c=>3}, {b=>20, c=>30, d=>40}); # => {b=>2, c=>3}# filtering keys using a coderefmy %subset = hash_subset ({a=>1, b=>2, c=>3}, sub {$_[0] =~ /[bc]/}); # => (b=>2, c=>3)my $subset = hashref_subset({a=>1, b=>2, c=>3}, sub {$_[0] =~ /[bc]/}); # => {b=>2, c=>3}# multiple filters: array, hash, coderefmy %subset = hash_subset ({a=>1, b=>2, c=>3, d=>4}, {c=>1}, [qw/b/], sub {$_[0] =~ /[bcd]/}); # => (b=>2, c=>3, d=>4)my $subset = hashref_subset({a=>1, b=>2, c=>3, d=>4}, {c=>1}, [qw/b/], sub {$_[0] =~ /[bcd]/}); # => {b=>2, c=>3, d=>4}# excluding keysmy %subset = hash_subset_without ({a=>1, b=>2, c=>3}, ['b','c','d']); # => (a=>1)my $subset = hashref_subset_without({a=>1, b=>2, c=>3}, ['b','c','d']); # => {a=>1}
A use case is when you use hash arguments:
sub func1 { my %args = @_; # known arguments: foo, bar, baz ...}sub func2 { my %args = @_; # known arguments: all func1 arguments as well as qux, quux # call func1 with all arguments passed to us my $res = func1(hash_subset(\%args, [qw/foo bar baz/])); # postprocess result ...}
If you useRinci metadata in your code, this will come in handy, for example:
my %common_args = ( foo => {...}, bar => {...}, baz => {...},);$SPEC{func1} = { v => 1.1, args => { %common_args, },};sub func1 { my %args = @_; ...}$SPEC{func2} = { v => 1.1, args => { %common_args, # func2 supports all func1 arguments plus a couple of others qux => { ... }, quux => { ... }, },};sub func2 { my %args = @_; # call func1 with all arguments passed to us my $res = func1(hash_subset(\%args, $SPEC{func1}{args})); # postprocess result ...}
Merging subset to another hash:
my %target = (a=>1, b=>2);merge_hash_subset(\%target, {foo=>1, bar=>2, baz=>3}, qr/ba/); # %target becomes (a=>1, b=>2, bar=>2, baz=>3)merge_hash_subset_without(\%target, {foo=>1, bar=>2, baz=>3}, qr/ba/); # %target becomes (a=>1, b=>2, foo=>1)
DESCRIPTION
Keywords: hash arguments, hash picking, hash grep, hash filtering, hash merging
FUNCTIONS
None exported by default.
hash_subset
Usage:
my %subset = hash_subset (\%hash, @keys_srcs);my $subset = hashref_subset(\%hash, @keys_srcs);
Where each @keys_src element can either be an arrayref, a hashref, a Regexp object, or a coderef. Coderef will be called with args($key, $value) and return true when key should be included.
Produce subset of%hash
, returning the subset hash (or hashref, in the case ofhashref_subset
function).
Perl lets you produce a hash subset using the hash slice notation:
my %subset = %hash{"b","c","d"};
The difference withhash_subset
is: 1) hash slice is only available since perl 5.20 (in previous versions, only array slice is available); 2) when the key does not exist in the array, perl will create it for you withundef
as the value:
my %hash = (a=>1, b=>2, c=>3);my %subset = %hash{"b","c","d"}; # => (b=>2, c=>3, d=>undef)
So basicallyhash_subset
is equivalent to:
my %subset = %hash{grep {exists $hash{$_}} "b","c","d"}; # => (b=>2, c=>3)
and available for perl earlier than 5.20. In addition to that, hash_subset() accepts arrayref & Regexp object as well as hashref/coderef, and several of them.
hashref_subset
See"hash_subset".
hash_subset_without
Like"hash_subset", but reverses the logic: will create subset that only includes keys not in the specified arrays/hashes/Regexps/coderefs.
hashref_subset_without
merge_hash_subset
Usage:
merge_hash_subset (\%h1, \%h2, @keys_src);merge_overwrite_hash_subset(\%h1, \%h2, @keys_src);merge_ignore_hash_subset (\%h1, \%h2, @keys_src);
merge_hash_subset
selects a subset of hash%h2
(using@keys_src
, just like in"hash_subset") and merge the subset to hash%h1
. This is basically a convenience shortcut for:
my %subset = hash_subset(\%h2, @keys_src);for my $key (keys %subset) { die "Duplicate key when merging subset: $key" if exists $h1{$key]; $h1{$key} = $subset{$key};}
whilemerge_overwrite_hash_subset
does something like this:
my %subset = hash_subset(\%h2, @keys_src);for my $key (keys %subset) { $h1{$key} = $subset{$key};}
andmerge_ignore_hash_subset
does something like this:
my %subset = hash_subset(\%h2, @keys_src);for my $key (keys %subset) { next if exists $h1{$key}; $h1{$key} = $subset{$key};}
merge_overwrite_hash_subset
merge_ignore_hash_subset
merge_hash_subset_without
Usage:
merge_hash_subset_without (\%h1, \%h2, @keys_src);merge_overwrite_hash_subset_without(\%h1, \%h2, @keys_src);merge_ignore_hash_subset_without (\%h1, \%h2, @keys_src);
These are like"merge_hash_subset","merge_overwrite_hash_subset", and"merge_ignore_hash_subset" except these routines will merge subset from%h2
that donot contain keys specified by@keys_src
.
merge_overwrite_hash_subset_without
See"merge_hash_subset_without".
merge_ignore_hash_subset_without
See"merge_hash_subset_without".
HOMEPAGE
Please visit the project's homepage athttps://metacpan.org/release/Hash-Subset.
SOURCE
Source repository is athttps://github.com/perlancar/perl-Hash-Subset.
SEE ALSO
Hash::MoreUtils provides various ways to create hash subset ("slice") through itsslice_*
functions. It does not provide way to specify subset keys via the keys of%another_hash
, but that can be done trivially usingkeys %another_hash
. Hash::Subset is currently more lightweight than Hash::MoreUtils.
Tie::Subset::Hash to create a tied version of a hash subset (a "view" of a subset of a hash).
Hash::Util::Pick also allows you to create a hash subset by specifying the wanted keys in a list or via filtering using a coderef. This XS module should perhaps be preferred over Hash::Subset for its performance, but there are some cases where you cannot use XS modules.
See some benchmarks inBencher::Scenarios::HashPicking.
AUTHOR
perlancar <perlancar@cpan.org>
CONTRIBUTING
To contribute, you can send patches by email/via RT, or send pull requests on GitHub.
Most of the time, you don't need to build the distribution yourself. You can simply modify the code, then test via:
% prove -l
If you want to build the distribution (e.g. to try to install it locally on your system), you can installDist::Zilla,Dist::Zilla::PluginBundle::Author::PERLANCAR, and sometimes one or two other Dist::Zilla plugin and/or Pod::Weaver::Plugin. Any additional steps required beyond that are considered a bug and can be reported to me.
COPYRIGHT AND LICENSE
This software is copyright (c) 2022, 2020, 2019 by perlancar <perlancar@cpan.org>.
This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.
BUGS
Please report any bugs or feature requests on the bugtracker websitehttps://rt.cpan.org/Public/Dist/Display.html?Name=Hash-Subset
When submitting a bug or request, please include a test-file or a patch to an existing test-file that illustrates the bug or desired feature.
Module Install Instructions
To install Hash::Subset, copy and paste the appropriate command in to your terminal.
cpanm Hash::Subset
perl -MCPAN -e shellinstall Hash::Subset
For more information on module installation, please visitthe detailed CPAN module installation guide.