NAME Acme::CPANModules::OrderedHash - List of modules that provide ordered hash data type VERSION This document describes version 0.003 of Acme::CPANModules::OrderedHash (from Perl distribution Acme-CPANModules-OrderedHash), released on 2023-10-06. SYNOPSIS To run benchmark with default option: % bencher --cpanmodules-module OrderedHash To run module startup overhead benchmark: % bencher --module-startup --cpanmodules-module OrderedHash For more options (dump scenario, list/include/exclude/add participants, list/include/exclude/add datasets, etc), see bencher or run "bencher --help". DESCRIPTION When you ask a Perl's hash for the list of keys, the answer comes back unordered. In fact, Perl explicitly randomizes the order of keys it returns everytime. The random ordering is a (security) feature, not a bug. However, sometimes you want to know the order of insertion. These modules provide you with an ordered hash; most of them implement it by recording the order of insertion of keys in an additional array. Other related modules: Tie::SortHash - will automatically sort keys when you call keys(), values(), each(). But this module does not maintain insertion order. ACME::CPANMODULES ENTRIES Tie::IxHash Author: CHORNY <https://metacpan.org/author/CHORNY> Hash::Ordered Author: DAGOLDEN <https://metacpan.org/author/DAGOLDEN> Tie::Hash::Indexed Author: MHX <https://metacpan.org/author/MHX> Provides two interfaces: tied hash and OO. Tie::LLHash Author: XAERXESS <https://metacpan.org/author/XAERXESS> Tie::StoredOrderHash Author: TFM <https://metacpan.org/author/TFM> Array::OrdHash Author: WOWASURIN <https://metacpan.org/author/WOWASURIN> Provide something closest to PHP's associative array, where you can refer elements by key or by numeric index, and insertion order is remembered. List::Unique::DeterministicOrder Author: SLAFFAN <https://metacpan.org/author/SLAFFAN> Provide a list, not hash. BENCHMARKED MODULES Version numbers shown below are the versions used when running the sample benchmark. Tie::IxHash 1.23 Hash::Ordered 0.014 Tie::Hash::Indexed 0.08 Tie::LLHash 1.004 Tie::StoredOrderHash 0.22 Array::OrdHash 1.03 List::Unique::DeterministicOrder 0.004 BENCHMARK PARTICIPANTS * Tie::IxHash (perl_code) Tie::IxHash * Hash::Ordered (perl_code) Hash::Ordered * Tie::Hash::Indexed (perl_code) Tie::Hash::Indexed * Tie::LLHash (perl_code) Tie::LLHash * Tie::StoredOrderHash (perl_code) Tie::StoredOrderHash * Array::OrdHash (perl_code) Array::OrdHash * List::Unique::DeterministicOrder (perl_code) [no_iterate] List::Unique::DeterministicOrder BENCHMARK DATASETS * insert 1000 pairs * insert 1000 pairs + delete * insert 1000 pairs + return keys 100 times * insert 1000 pairs + iterate 10 times BENCHMARK SAMPLE RESULTS Sample benchmark #1 Run on: perl: *v5.38.0*, CPU: *Intel(R) Core(TM) i5-7200U CPU @ 2.50GHz (2 cores)*, OS: *GNU/Linux Ubuntu version 20.04*, OS kernel: *Linux version 5.4.0-91-generic*. Benchmark command (default options): % bencher --cpanmodules-module OrderedHash Result formatted as table (split, part 1 of 4): #table1# {dataset=>"insert 1000 pairs"} +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ | participant | p_tags | rate (/s) | time (ms) | pct_faster_vs_slowest | pct_slower_vs_fastest | errors | samples | +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ | Tie::StoredOrderHash | | 360 | 2.78 | 0.00% | 197.99% | 1.1e-06 | 20 | | Tie::LLHash | | 380 | 2.6 | 6.60% | 179.55% | 1.3e-05 | 20 | | Array::OrdHash | | 540 | 1.9 | 49.87% | 98.83% | 3.2e-06 | 20 | | Tie::Hash::Indexed | | 700 | 2 | 81.84% | 63.87% | 7.2e-05 | 22 | | Tie::IxHash | | 676 | 1.48 | 87.57% | 58.87% | 9.9e-07 | 20 | | Hash::Ordered | | 884 | 1.13 | 145.25% | 21.50% | 1.1e-06 | 21 | | List::Unique::DeterministicOrder | no_iterate | 1070 | 0.931 | 197.99% | 0.00% | 8e-07 | 21 | +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ The above result formatted in Benchmark.pm style: Rate T:S T:L TH:I A:O T:I H:O LU:D no_iterate T:S 360/s -- -6% -28% -31% -46% -59% -66% T:L 380/s 6% -- -23% -26% -43% -56% -64% TH:I 700/s 38% 30% -- -5% -26% -43% -53% A:O 540/s 46% 36% 5% -- -22% -40% -51% T:I 676/s 87% 75% 35% 28% -- -23% -37% H:O 884/s 146% 130% 76% 68% 30% -- -17% LU:D no_iterate 1070/s 198% 179% 114% 104% 58% 21% -- Legends: A:O : p_tags= participant=Array::OrdHash H:O : p_tags= participant=Hash::Ordered LU:D no_iterate: p_tags=no_iterate participant=List::Unique::DeterministicOrder T:I : p_tags= participant=Tie::IxHash T:L : p_tags= participant=Tie::LLHash T:S : p_tags= participant=Tie::StoredOrderHash TH:I : p_tags= participant=Tie::Hash::Indexed The above result presented as chart: Result formatted as table (split, part 2 of 4): #table2# {dataset=>"insert 1000 pairs + delete"} +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ | participant | p_tags | rate (/s) | time (ms) | pct_faster_vs_slowest | pct_slower_vs_fastest | errors | samples | +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ | Tie::IxHash | | 17 | 58.8 | 0.00% | 3799.19% | 4.1e-05 | 20 | | Tie::StoredOrderHash | | 200 | 5 | 1070.77% | 233.04% | 1.1e-05 | 20 | | Tie::LLHash | | 220 | 4.6 | 1191.67% | 201.87% | 1.4e-05 | 21 | | Array::OrdHash | | 279 | 3.59 | 1537.64% | 138.10% | 2.8e-06 | 21 | | Hash::Ordered | | 370 | 2.7 | 2087.76% | 78.23% | 4.9e-06 | 20 | | List::Unique::DeterministicOrder | no_iterate | 604 | 1.66 | 3450.01% | 9.84% | 5.1e-07 | 20 | | Tie::Hash::Indexed | | 663 | 1.51 | 3799.19% | 0.00% | 6e-07 | 20 | +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ The above result formatted in Benchmark.pm style: Rate T:I T:S T:L A:O H:O LU:D no_iterate TH:I T:I 17/s -- -91% -92% -93% -95% -97% -97% T:S 200/s 1076% -- -8% -28% -46% -66% -69% T:L 220/s 1178% 8% -- -21% -41% -63% -67% A:O 279/s 1537% 39% 28% -- -24% -53% -57% H:O 370/s 2077% 85% 70% 32% -- -38% -44% LU:D no_iterate 604/s 3442% 201% 177% 116% 62% -- -9% TH:I 663/s 3794% 231% 204% 137% 78% 9% -- Legends: A:O : p_tags= participant=Array::OrdHash H:O : p_tags= participant=Hash::Ordered LU:D no_iterate: p_tags=no_iterate participant=List::Unique::DeterministicOrder T:I : p_tags= participant=Tie::IxHash T:L : p_tags= participant=Tie::LLHash T:S : p_tags= participant=Tie::StoredOrderHash TH:I : p_tags= participant=Tie::Hash::Indexed The above result presented as chart: Result formatted as table (split, part 3 of 4): #table3# {dataset=>"insert 1000 pairs + iterate 10 times"} +----------------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ | participant | rate (/s) | time (ms) | pct_faster_vs_slowest | pct_slower_vs_fastest | errors | samples | +----------------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ | Tie::LLHash | 45 | 22 | 0.00% | 206.84% | 4.1e-05 | 21 | | Tie::StoredOrderHash | 46 | 21.7 | 2.02% | 200.75% | 9.4e-06 | 20 | | Array::OrdHash | 51.4 | 19.5 | 13.93% | 169.32% | 1.4e-05 | 21 | | Tie::IxHash | 65.1 | 15.4 | 44.41% | 112.47% | 9.4e-06 | 20 | | Tie::Hash::Indexed | 97.5 | 10.3 | 116.20% | 41.93% | 9.3e-06 | 20 | | Hash::Ordered | 140 | 7.2 | 206.84% | 0.00% | 6.4e-05 | 20 | +----------------------+-----------+-----------+-----------------------+-----------------------+---------+---------+ The above result formatted in Benchmark.pm style: Rate T:L T:S A:O T:I TH:I H:O T:L 45/s -- -1% -11% -29% -53% -67% T:S 46/s 1% -- -10% -29% -52% -66% A:O 51.4/s 12% 11% -- -21% -47% -63% T:I 65.1/s 42% 40% 26% -- -33% -53% TH:I 97.5/s 113% 110% 89% 49% -- -30% H:O 140/s 205% 201% 170% 113% 43% -- Legends: A:O: participant=Array::OrdHash H:O: participant=Hash::Ordered T:I: participant=Tie::IxHash T:L: participant=Tie::LLHash T:S: participant=Tie::StoredOrderHash TH:I: participant=Tie::Hash::Indexed The above result presented as chart: Result formatted as table (split, part 4 of 4): #table4# {dataset=>"insert 1000 pairs + return keys 100 times"} +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+-----------+---------+ | participant | p_tags | rate (/s) | time (ms) | pct_faster_vs_slowest | pct_slower_vs_fastest | errors | samples | +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+-----------+---------+ | Tie::StoredOrderHash | | 10.7 | 93.2 | 0.00% | 845.55% | 5.8e-05 | 20 | | Tie::LLHash | | 12 | 86 | 8.76% | 769.40% | 0.00011 | 20 | | Array::OrdHash | | 14 | 70 | 32.35% | 614.46% | 0.0001 | 20 | | Tie::IxHash | | 16.2 | 61.8 | 50.70% | 527.45% | 5.4e-05 | 20 | | Tie::Hash::Indexed | | 24 | 41 | 124.79% | 320.64% | 0.0001 | 20 | | Hash::Ordered | | 78.3 | 12.8 | 629.18% | 29.67% | 4.2e-06 | 20 | | List::Unique::DeterministicOrder | no_iterate | 101 | 9.85 | 845.55% | 0.00% | 2.2e-06 | 22 | +----------------------------------+------------+-----------+-----------+-----------------------+-----------------------+-----------+---------+ The above result formatted in Benchmark.pm style: Rate T:S T:L A:O T:I TH:I H:O LU:D no_iterate T:S 10.7/s -- -7% -24% -33% -56% -86% -89% T:L 12/s 8% -- -18% -28% -52% -85% -88% A:O 14/s 33% 22% -- -11% -41% -81% -85% T:I 16.2/s 50% 39% 13% -- -33% -79% -84% TH:I 24/s 127% 109% 70% 50% -- -68% -75% H:O 78.3/s 628% 571% 446% 382% 220% -- -23% LU:D no_iterate 101/s 846% 773% 610% 527% 316% 29% -- Legends: A:O : p_tags= participant=Array::OrdHash H:O : p_tags= participant=Hash::Ordered LU:D no_iterate: p_tags=no_iterate participant=List::Unique::DeterministicOrder T:I : p_tags= participant=Tie::IxHash T:L : p_tags= participant=Tie::LLHash T:S : p_tags= participant=Tie::StoredOrderHash TH:I : p_tags= participant=Tie::Hash::Indexed The above result presented as chart: Sample benchmark #2 Benchmark command (benchmarking module startup overhead): % bencher --cpanmodules-module OrderedHash --module-startup Result formatted as table: #table5# +----------------------------------+-----------+-------------------+-----------------------+-----------------------+---------+---------+ | participant | time (ms) | mod_overhead_time | pct_faster_vs_slowest | pct_slower_vs_fastest | errors | samples | +----------------------------------+-----------+-------------------+-----------------------+-----------------------+---------+---------+ | List::Unique::DeterministicOrder | 16.2 | 8.5 | 0.00% | 110.26% | 7.7e-06 | 20 | | Hash::Ordered | 15.9 | 8.2 | 1.50% | 107.15% | 6e-06 | 20 | | Tie::Hash::Indexed | 15.5 | 7.8 | 4.57% | 101.06% | 4.9e-06 | 22 | | Array::OrdHash | 15 | 7.3 | 7.86% | 94.94% | 1.1e-05 | 20 | | Tie::IxHash | 14.9 | 7.2 | 8.49% | 93.81% | 9e-06 | 20 | | Tie::LLHash | 13.6 | 5.9 | 18.93% | 76.79% | 7.4e-06 | 20 | | Tie::StoredOrderHash | 10.7 | 3 | 51.34% | 38.93% | 5.9e-06 | 20 | | perl -e1 (baseline) | 7.7 | 0 | 110.26% | 0.00% | 3.4e-05 | 20 | +----------------------------------+-----------+-------------------+-----------------------+-----------------------+---------+---------+ The above result formatted in Benchmark.pm style: Rate LU:D H:O TH:I A:O T:I T:L T:S perl -e1 (baseline) LU:D 61.7/s -- -1% -4% -7% -8% -16% -33% -52% H:O 62.9/s 1% -- -2% -5% -6% -14% -32% -51% TH:I 64.5/s 4% 2% -- -3% -3% -12% -30% -50% A:O 66.7/s 7% 6% 3% -- 0% -9% -28% -48% T:I 67.1/s 8% 6% 4% 0% -- -8% -28% -48% T:L 73.5/s 19% 16% 13% 10% 9% -- -21% -43% T:S 93.5/s 51% 48% 44% 40% 39% 27% -- -28% perl -e1 (baseline) 129.9/s 110% 106% 101% 94% 93% 76% 38% -- Legends: A:O: mod_overhead_time=7.3 participant=Array::OrdHash H:O: mod_overhead_time=8.2 participant=Hash::Ordered LU:D: mod_overhead_time=8.5 participant=List::Unique::DeterministicOrder T:I: mod_overhead_time=7.2 participant=Tie::IxHash T:L: mod_overhead_time=5.9 participant=Tie::LLHash T:S: mod_overhead_time=3 participant=Tie::StoredOrderHash TH:I: mod_overhead_time=7.8 participant=Tie::Hash::Indexed perl -e1 (baseline): mod_overhead_time=0 participant=perl -e1 (baseline) The above result presented as chart: To display as an interactive HTML table on a browser, you can add option "--format html+datatables". BENCHMARK NOTES Hash::Ordered has strong performance in iterating and returning keys, while List::Unique::DeterministicOrder is strong in insertion and deletion (or Tie::Hash::Indexed if you're looking for actual hash type). FAQ What is an Acme::CPANModules::* module? An Acme::CPANModules::* module, like this module, contains just a list of module names that share a common characteristics. It is a way to categorize modules and document CPAN. See Acme::CPANModules for more details. What are ways to use this Acme::CPANModules module? Aside from reading this Acme::CPANModules module's POD documentation, you can install all the listed modules (entries) using cpanm-cpanmodules script (from App::cpanm::cpanmodules distribution): % cpanm-cpanmodules -n OrderedHash Alternatively you can use the cpanmodules CLI (from App::cpanmodules distribution): % cpanmodules ls-entries OrderedHash | cpanm -n or Acme::CM::Get: % perl -MAcme::CM::Get=OrderedHash -E'say $_->{module} for @{ $LIST->{entries} }' | cpanm -n or directly: % perl -MAcme::CPANModules::OrderedHash -E'say $_->{module} for @{ $Acme::CPANModules::OrderedHash::LIST->{entries} }' | cpanm -n This Acme::CPANModules module contains benchmark instructions. You can run a benchmark for some/all the modules listed in this Acme::CPANModules module using the bencher CLI (from Bencher distribution): % bencher --cpanmodules-module OrderedHash This Acme::CPANModules module also helps lcpan produce a more meaningful result for "lcpan related-mods" command when it comes to finding related modules for the modules listed in this Acme::CPANModules module. See App::lcpan::Cmd::related_mods for more details on how "related modules" are found. HOMEPAGE Please visit the project's homepage at <https://metacpan.org/release/Acme-CPANModules-OrderedHash>. SOURCE Source repository is at <https://github.com/perlancar/perl-Acme-CPANModules-OrderedHash>. SEE ALSO Acme::CPANModules::HashUtilities Acme::CPANModules - about the Acme::CPANModules namespace cpanmodules - CLI tool to let you browse/view the lists 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 install Dist::Zilla, Dist::Zilla::PluginBundle::Author::PERLANCAR, Pod::Weaver::PluginBundle::Author::PERLANCAR, and sometimes one or two other Dist::Zilla- and/or Pod::Weaver plugins. Any additional steps required beyond that are considered a bug and can be reported to me. COPYRIGHT AND LICENSE This software is copyright (c) 2023 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 website <https://rt.cpan.org/Public/Dist/Display.html?Name=Acme-CPANModules-Orde redHash> 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.