diff --git a/benchmark/encoder.rb b/benchmark/encoder.rb index 39d3e906..acc5fa07 100644 --- a/benchmark/encoder.rb +++ b/benchmark/encoder.rb @@ -17,7 +17,6 @@ def implementations(ruby_obj) state = JSON::State.new(JSON.dump_default_options) { - json_state: ["json (reuse)", proc { state.generate(ruby_obj) }], json: ["json", proc { JSON.generate(ruby_obj) }], oj: ["oj", proc { Oj.dump(ruby_obj) }], } @@ -58,27 +57,24 @@ def benchmark_encoding(benchmark_name, ruby_obj, check_expected: true, except: [ # NB: Notes are based on ruby 3.3.4 (2024-07-09 revision be1089c8ec) +YJIT [arm64-darwin23] # On the first two micro benchmarks, the limitting factor is the fixed cost of initializing the -# generator state. Since `JSON.generate` now lazily allocate the `State` object we're now ~10% faster +# generator state. Since `JSON.generate` now lazily allocate the `State` object we're now ~10-20% faster # than `Oj.dump`. benchmark_encoding "small mixed", [1, "string", { a: 1, b: 2 }, [3, 4, 5]] benchmark_encoding "small nested array", [[1,2,3,4,5]]*10 - -# On small hash specifically, we're just on par with `Oj.dump`. Would be worth investigating why -# Hash serialization doesn't perform as well as other types. benchmark_encoding "small hash", { "username" => "jhawthorn", "id" => 123, "event" => "wrote json serializer" } -# On string encoding we're ~20% faster when dealing with mostly ASCII, but ~10% slower when dealing -# with mostly multi-byte characters. This is a tradeoff. -benchmark_encoding "mixed utf8", ([("a" * 5000) + "€" + ("a" * 5000)] * 500), except: %i(json_state) -benchmark_encoding "mostly utf8", ([("€" * 3333)] * 500), except: %i(json_state) +# On string encoding we're ~20% faster when dealing with mostly ASCII, but ~50% slower when dealing +# with mostly multi-byte characters. There's likely some gains left to be had in multi-byte handling. +benchmark_encoding "mixed utf8", ([("a" * 5000) + "€" + ("a" * 5000)] * 500) +benchmark_encoding "mostly utf8", ([("€" * 3333)] * 500) # On these benchmarks we perform well, we're on par or better. benchmark_encoding "integers", (1_000_000..1_001_000).to_a, except: %i(json_state) -benchmark_encoding "activitypub.json", JSON.load_file("#{__dir__}/data/activitypub.json"), except: %i(json_state) -benchmark_encoding "citm_catalog.json", JSON.load_file("#{__dir__}/data/citm_catalog.json"), except: %i(json_state) +benchmark_encoding "activitypub.json", JSON.load_file("#{__dir__}/data/activitypub.json") +benchmark_encoding "citm_catalog.json", JSON.load_file("#{__dir__}/data/citm_catalog.json") -# On twitter.json we're still about 10% slower, this is worth investigating. -benchmark_encoding "twitter.json", JSON.load_file("#{__dir__}/data/twitter.json"), except: %i(json_state) +# On twitter.json we're still about 6% slower, this is worth investigating. +benchmark_encoding "twitter.json", JSON.load_file("#{__dir__}/data/twitter.json") # This benchmark spent the overwhelming majority of its time in `ruby_dtoa`. We rely on Ruby's implementation # which uses a relatively old version of dtoa.c from David M. Gay. @@ -89,8 +85,8 @@ def benchmark_encoding(benchmark_name, ruby_obj, check_expected: true, except: [ # but all these are implemented in C++11 or newer, making it hard if not impossible to include them. # Short of a pure C99 implementation of these newer algorithms, there isn't much that can be done to match # Oj speed without losing precision. -benchmark_encoding "canada.json", JSON.load_file("#{__dir__}/data/canada.json"), check_expected: false, except: %i(json_state) +benchmark_encoding "canada.json", JSON.load_file("#{__dir__}/data/canada.json"), check_expected: false # We're about 10% faster when `to_json` calls are involved, but this wasn't particularly optimized, there might be # opportunities here. -benchmark_encoding "many #to_json calls", [{object: Object.new, int: 12, float: 54.3, class: Float, time: Time.now, date: Date.today}] * 20, except: %i(json_state) +benchmark_encoding "many #to_json calls", [{object: Object.new, int: 12, float: 54.3, class: Float, time: Time.now, date: Date.today}] * 20 diff --git a/benchmark/parser.rb b/benchmark/parser.rb index 4425728a..6952f3c3 100644 --- a/benchmark/parser.rb +++ b/benchmark/parser.rb @@ -28,27 +28,26 @@ def benchmark_parsing(name, json_output) # NB: Notes are based on ruby 3.3.4 (2024-07-09 revision be1089c8ec) +YJIT [arm64-darwin23] -# Oj::Parser is very significanly faster (1.80x) on the nested array benchmark. -benchmark_parsing "small nested array", JSON.dump([[1,2,3,4,5]]*10) - -# Oj::Parser is significanly faster (~1.5x) on the next 4 benchmarks in large part because its +# Oj::Parser is significanly faster (~1.3x) on the next 3 micro-benchmarks in large part because its # cache is persisted across calls. That's not something we can do with the current API, we'd # need to expose a stateful API as well, but that's no really desirable. -# Other than that we're faster than regular `Oj.load` by a good margin. +# Other than that we're faster than regular `Oj.load` by a good margin (between 1.3x and 2.4x). +benchmark_parsing "small nested array", JSON.dump([[1,2,3,4,5]]*10) benchmark_parsing "small hash", JSON.dump({ "username" => "jhawthorn", "id" => 123, "event" => "wrote json serializer" }) - benchmark_parsing "test from oj", <