Skip to content

Commit

Permalink
📝 New benchmarks
Browse files Browse the repository at this point in the history
Ruby 3.3.5:
- memery
  • Loading branch information
pboling committed Sep 17, 2024
1 parent 360c1f6 commit dc24407
Show file tree
Hide file tree
Showing 3 changed files with 61 additions and 46 deletions.
54 changes: 27 additions & 27 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -114,37 +114,37 @@ For more usage details, see our detailed [documentation](#documentation).

Benchmarks are run in GitHub Actions, and the tables below are updated with every code change. **Values >1.00x represent how much _slower_ each gem’s memoized value retrieval is than the latest commit of `MemoWise`**, according to [`benchmark-ips`](https://github.com/evanphx/benchmark-ips) (2.11.0).

Results using Ruby 3.3.2:

|Method arguments|`Dry::Core` (1.0.1)|`Memery` (alt_memery|`Memoist` (memoist3|
|--|--|--|--|
|`()` (none)|0.44x|10.91x|3.73x|
|`(a)`|0.87x|6.65x|14.31x|
|`(a, b)`|0.69x|4.80x|9.86x|
|`(a:)`|0.98x|10.12x|19.52x|
|`(a:, b:)`|0.71x|8.49x|17.15x|
|`(a, b:)`|0.70x|8.03x|13.98x|
|`(a, *args)`|0.80x|2.09x|3.62x|
|`(a:, **kwargs)`|0.63x|2.46x|6.36x|
|`(a, *args, b:, **kwargs)`|0.68x|1.69x|4.03x|
Results using Ruby 3.3.5:

| Method arguments | `alt_memery` (2.1.0) | `dry-core` \* (1.0.1) | `memery` (1.6.0) | `memoist3` (1.0.0) |
|----------------------------|----------------------|-----------------------|------------------|--------------------|
| `()` (none) | 11.86x | 0.45x | 5.23x | 3.72x |
| `(a)` | 6.60x | 0.84x | 5.70x | 13.84x |
| `(a, b)` | 5.02x | 0.72x | 4.15x | 10.53x |
| `(a:)` | 9.83x | 0.83x | 5.94x | 17.24x |
| `(a:, b:)` | 7.66x | 0.70x | 4.66x | 16.24x |
| `(a, b:)` | 7.87x | 0.70x | 4.64x | 13.38x |
| `(a, *args)` | 2.08x | 0.78x | 1.57x | 3.58x |
| `(a:, **kwargs)` | 2.46x | 0.59x | 1.78x | 6.84x |
| `(a, *args, b:, **kwargs)` | 1.67x | 0.67x | 1.12x | 4.10x |

\* `Dry::Core`
[may cause incorrect behavior caused by hash collisions](https://github.com/dry-rb/dry-core/issues/63).

Results using Ruby 2.7.8 (either because these gems raise errors in Ruby 3.x,
or due to namespace collisions with more modern forks run against Ruby 3.x):

|Method arguments|`DDMemoize` (1.0.0)|`Memery` (memery|`Memoist` (memoist|`Memoized` (1.1.1)|`Memoizer` (1.0.3)|
|--|--|--|--|--|--|
|`()` (none)|20.10x|3.79x|2.35x|22.22x|3.05x|
|`(a)`|17.03x|8.70x|12.39x|17.75x|11.06x|
|`(a, b)`|15.26x|8.41x|11.13x|15.27x|10.06x|
|`(a:)`|22.28x|12.52x|18.17x|20.59x|16.45x|
|`(a:, b:)`|10.09x|5.77x|9.15x|10.83x|11.99x|
|`(a, b:)`|20.01x|11.25x|16.88x|18.26x|15.34x|
|`(a, *args)`|2.98x|1.56x|2.27x|3.04x|1.96x|
|`(a:, **kwargs)`|2.73x|1.55x|2.20x|2.49x|2.10x|
|`(a, *args, b:, **kwargs)`|2.16x|1.16x|1.85x|1.99x|1.74x|
Results using Ruby 2.7.8 (because these gems raise errors in Ruby 3.x):

| Method arguments | `ddmemoize` (1.0.0) | `memoist` (0.16.2) | `memoized` (1.1.1) | `memoizer` (1.0.3) |
|----------------------------|---------------------|--------------------|--------------------|--------------------|
| `()` (none) | 19.77x | 2.41x | 21.53x | 3.08x |
| `(a)` | 16.41x | 11.80x | 17.03x | 10.38x |
| `(a, b)` | 15.09x | 11.17x | 15.21x | 9.96x |
| `(a:)` | 20.91x | 17.19x | 18.48x | 15.66x |
| `(a:, b:)` | 20.48x | 17.30x | 18.74x | 16.18x |
| `(a, b:)` | 19.17x | 16.03x | 16.93x | 15.57x |
| `(a, *args)` | 2.91x | 2.08x | 2.90x | 1.81x |
| `(a:, **kwargs)` | 2.61x | 2.20x | 2.61x | 2.18x |
| `(a, *args, b:, **kwargs)` | 2.13x | 1.79x | 1.99x | 1.71x |


You can run benchmarks yourself with:

Expand Down
11 changes: 8 additions & 3 deletions benchmarks/Gemfile
Original file line number Diff line number Diff line change
Expand Up @@ -10,18 +10,23 @@ gem "benchmark", "0.3.0"
gem "benchmark-ips", "2.14.0"
gem "gem_bench", "2.0.1"

# NOTE: Regarding `require: false` below
# 1. GitHub version of MemoWise and the local source of MemoWise, share a namespace
# 2. memery & alt_memery share the namespace Memery
# 3. memoist & memoist3 share the namespace Memoist, and also share a load path for their version.rb files.
# This means we must `require: false` in `benchmarks/Gemfile` all, or all but one, of each of these duplicates,
# or we take care to only load them in discrete Ruby versions,
# to avoid a namespace collision before re-namespacing duplicates
if RUBY_VERSION > "3"
gem "alt_memery", "2.1.0", require: false
gem "dry-core", "1.0.1"
gem "memoist3", "1.0.0", require: false
gem "memoist2", "0.3.0"
gem "memery", "1.6.0"
else
gem "ddmemoize", "1.0.0"
gem "memoist", "0.16.2"
gem "memoist2", "0.3.0"
gem "memoized", "1.1.1"
gem "memoizer", "1.0.3"
end

gem "memo_wise", github: "panorama-ed/memo_wise", branch: "main"
gem "memo_wise", github: "panorama-ed/memo_wise", branch: "main", require: false
42 changes: 26 additions & 16 deletions benchmarks/benchmarks.rb
Original file line number Diff line number Diff line change
Expand Up @@ -5,14 +5,17 @@
require "benchmark/ips"
require "gem_bench/jersey"

# Constants used for temp file paths necessary to separate gem namespaces that would otherwise collide.
GITHUB_MAIN = "MemoWise_GitHubMain"
GITHUB_MAIN_BENCHMARK_NAME = "memo_wise-git-main"
LOCAL_BENCHMARK_NAME = "memo_wise-local"

# Constants used for temp file paths necessary to separate gem namespaces that would otherwise collide.
# 1. GitHub version of MemoWise and the local source of MemoWise, share a namespace
# 2. memery & alt_memery share the namespace Memery
# 3. memoist, memoist2, & memoist3 share the namespace Memoist
# 3. memoist & memoist3 share the namespace Memoist, and also share a load path for their version.rb files.
# This means we must `require: false` in `benchmarks/Gemfile` all, or all but one, of each of these duplicates,
# or we take care to only load them in discrete Ruby versions,
# to avoid a namespace collision before re-namespacing duplicates
re_namespaced_gems = [
GemBench::Jersey.new(
gem_name: "memo_wise",
Expand Down Expand Up @@ -43,9 +46,19 @@
activation_code: "extend MemoistThree",
memoization_method: :memoize,
},
)
),
GemBench::Jersey.new(
gem_name: "memoist",
trades: {
"Memoist" => "MemoistOne"
},
metadata: {
activation_code: "extend MemoistOne",
memoization_method: :memoize,
},
),
].each do |re_namespaced_gem|
re_namespaced_gem.doff_and_don
re_namespaced_gem.doff_and_don # Copies, re-namespaces, and requires each gem.
end

# Here we require memo_wise gem from the `pwd` source.
Expand All @@ -54,18 +67,14 @@
require_relative "../lib/memo_wise"

# Some gems do not yet work in Ruby 3 so we only require them if they're loaded
# in the Gemfile.
%w[memery memoist memoized memoizer ddmemoize dry-core].
# in the Gemfile. Gems re-namespaced by GemBench::Jersey will have already been loaded by now.
%w[memery memoist2 memoized memoizer ddmemoize dry-core].
each { |gem| require gem if Gem.loaded_specs.key?(gem) }

# alt_memery gem is a fork of memery, and uses the exact same namespace.
# This means we can't run them both at the same time, so we'll run memery on Ruby 2.7, and alt_memery on Ruby 3+.
# Similarly, memoist3 is a fork of memoist with Ruby 3 fixes.
# The Memoizable module from dry-core needs to be required manually.
# Some Gems Have Modules Which Need To Be Required Manually:
# 1. `dry-core` => Memoizable
{
# file_path => gem_name
"memery" => "alt_memery",
"memoist" => "memoist3",
"dry/core/memoizable" => "dry-core",
}.
each { |file_path, gem_name| require file_path if Gem.loaded_specs.key?(gem_name) }
Expand Down Expand Up @@ -116,16 +125,16 @@ def benchmark_name
re_namespaced_gem.gem_name == "memo_wise" ? GITHUB_MAIN_BENCHMARK_NAME : re_namespaced_gem.gem_name,
) if re_namespaced_gem.required?
}.concat(
[
[
BenchmarkGem.new(MemoWise, "prepend MemoWise", :memo_wise, LOCAL_BENCHMARK_NAME),
(BenchmarkGem.new(DDMemoize, "DDMemoize.activate(self)", :memoize, "ddmemoize") if defined?(DDMemoize)),
(BenchmarkGem.new(Dry::Core, "include Dry::Core::Memoizable", :memoize, "dry-core") if defined?(Dry::Core)),
(BenchmarkGem.new(Memery, "include Memery", :memoize, "memery") if defined?(Memery)),
(BenchmarkGem.new(Memoist, "extend Memoist", :memoize, "memoist") if defined?(Memoist)),
(BenchmarkGem.new(Memoist2, "include Memoist2", :memoize, "memoist2") if defined?(Memoist2)),
# (BenchmarkGem.new(Memoist, "extend Memoist", :memoize, "memoist") if defined?(Memoist)),
(BenchmarkGem.new(Memoized, "include Memoized", :memoize, "memoized") if defined?(Memoized)),
(BenchmarkGem.new(Memoizer, "include Memoizer", :memoize, "memoizer") if defined?(Memoizer))
])).compact.shuffle
]
)).compact.shuffle

puts "\nWill BENCHMARK_GEMS:\n\t#{BENCHMARK_GEMS.map(&:benchmark_name).join("\n\t")}\n"

Expand Down Expand Up @@ -297,6 +306,7 @@ def positional_splat_keyword_and_double_splat_args(a, *args, b:, **kwargs)
# "memoist (1.1.0): ()"
# We use this mapping to get a header of the form
# "`memoist` (1.1.0)
# "`memoist` (1.1.0)
gem_name_parts = benchmark_gem["name"].split
"`#{gem_name_parts[0]}` #{gem_name_parts[1][...-1]}"
end.join("|")
Expand Down

0 comments on commit dc24407

Please sign in to comment.