Skip to content

Commit f7b4edb

Browse files
authored
Add a few more benchmarks (#1449)
In order to better compare the performance between the existing v6 and the one with the [Fractions](https://github.com/danm-de/Fractions) I've been using these benchmarks. They are not _very polished_ (and I haven't had the time for the string-related stuff) but they do the trick. You don't have to merge them if you don't want to, but at least they are here (so I can reference the code). I'll probably add the results from my local runs in a comment (or more like a bunch of comments) later on (my previous results got wiped by the build script 😆) - but I was hoping that we could update the packages and add net9.0 to the list of targets first (there are [lots of a cool stuff](danm-de/Fractions#90) in net9). PS I'm not sure if that old `gh-action` would still work or not, but even adding the results by hand ([here?](https://github.com/angularsen/UnitsNet/tree/gh-pages/benchmarks/)) is an option. Here's what I did for the [Fractions](https://github.com/danm-de/Fractions/blob/master/benchmarks/Readme.md) (the `Readme.md` was 90% AI generated).
1 parent 428b58b commit f7b4edb

35 files changed

+1917
-81
lines changed
+82
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
// Licensed under MIT No Attribution, see LICENSE file at the root.
2+
// Copyright 2013 Andreas Gullberg Larsen ([email protected]). Maintained at https://github.com/angularsen/UnitsNet.
3+
4+
using System;
5+
using System.Collections;
6+
using System.Collections.Generic;
7+
using System.Linq;
8+
9+
namespace UnitsNet.Benchmark;
10+
11+
public static class BenchmarkHelpers
12+
{
13+
public static string[] GetRandomAbbreviations<TQuantity>(this Random random, UnitAbbreviationsCache abbreviations, int nbAbbreviations)
14+
{
15+
return random.GetItems(abbreviations.GetAllUnitAbbreviationsForQuantity(typeof(TQuantity)).ToArray(), nbAbbreviations);
16+
}
17+
18+
public static (TQuantity Quantity, TUnit Unit)[] GetRandomConversions<TQuantity, TUnit>(this Random random, double value, TUnit[] options,
19+
int nbConversions)
20+
where TQuantity : IQuantity<TUnit>
21+
where TUnit : struct, Enum
22+
{
23+
var quantities = GetRandomQuantities<TQuantity, TUnit>(random, value, options, nbConversions);
24+
TUnit[] units = random.GetItems(options, nbConversions);
25+
return quantities.Zip(units, (quantity, unit) => (quantity, unit)).ToArray();
26+
}
27+
28+
public static IEnumerable<TQuantity> GetRandomQuantities<TQuantity, TUnit>(this Random random, double value, TUnit[] units, int nbQuantities)
29+
where TQuantity : IQuantity<TUnit> where TUnit : struct, Enum
30+
{
31+
IEnumerable<TQuantity> quantities = random.GetItems(units, nbQuantities).Select(unit => (TQuantity)Quantity.From(value, unit));
32+
return quantities;
33+
}
34+
35+
#if !NET
36+
/// <summary>Creates an array populated with items chosen at random from the provided set of choices.</summary>
37+
/// <param name="random">The random number generator used to select items.</param>
38+
/// <param name="choices">The items to use to populate the array.</param>
39+
/// <param name="length">The length of array to return.</param>
40+
/// <typeparam name="T">The type of array.</typeparam>
41+
/// <exception cref="T:System.ArgumentException">
42+
/// <paramref name="choices" /> is empty.</exception>
43+
/// <exception cref="T:System.ArgumentNullException">
44+
/// <paramref name="choices" /> is <see langword="null" />.</exception>
45+
/// <exception cref="T:System.ArgumentOutOfRangeException">
46+
/// <paramref name="length" /> is not zero or a positive number.</exception>
47+
/// <returns>An array populated with random items.</returns>
48+
public static T[] GetItems<T>(this Random random, T[] choices, int length)
49+
{
50+
return GetItems<T>(random, new ReadOnlySpan<T>(choices), length);
51+
}
52+
53+
/// <summary>
54+
/// Generates an array of specified length with items chosen at random from the provided set of choices.
55+
/// </summary>
56+
/// <typeparam name="T">The type of the items.</typeparam>
57+
/// <param name="random">The random number generator used to select items.</param>
58+
/// <param name="choices">The set of items to choose from.</param>
59+
/// <param name="length">The length of the resulting array.</param>
60+
/// <returns>An array of randomly selected items.</returns>
61+
/// <exception cref="ArgumentException">Thrown when <paramref name="choices"/> is empty.</exception>
62+
public static T[] GetItems<T>(this Random random, ReadOnlySpan<T> choices, int length)
63+
{
64+
T[] array = new T[length];
65+
GetItems<T>(random, choices, array.AsSpan<T>());
66+
return array;
67+
}
68+
69+
/// <summary>Fills the elements of a specified span with items chosen at random from the provided set of choices.</summary>
70+
/// <param name="random">The random number generator used to select items.</param>
71+
/// <param name="choices">The items to use to populate the span.</param>
72+
/// <param name="destination">The span to be filled with items.</param>
73+
/// <typeparam name="T">The type of span.</typeparam>
74+
/// <exception cref="T:System.ArgumentException">
75+
/// <paramref name="choices" /> is empty.</exception>
76+
public static void GetItems<T>(this Random random, ReadOnlySpan<T> choices, Span<T> destination)
77+
{
78+
for (int index = 0; index < destination.Length; ++index)
79+
destination[index] = choices[random.Next(choices.Length)];
80+
}
81+
#endif
82+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
using System;
2+
using System.Collections.Generic;
3+
using BenchmarkDotNet.Attributes;
4+
using BenchmarkDotNet.Jobs;
5+
6+
namespace UnitsNet.Benchmark.Comparisons;
7+
8+
[MemoryDiagnoser]
9+
[ShortRunJob(RuntimeMoniker.Net48)]
10+
[ShortRunJob(RuntimeMoniker.Net80)]
11+
public class ComparisonBenchmarks
12+
{
13+
private static readonly Mass Tolerance = Mass.FromNanograms(1);
14+
15+
public static IEnumerable<object[]> Operands()
16+
{
17+
// equal value and unit
18+
yield return [Mass.From(42, Mass.BaseUnit), Mass.From(42, Mass.BaseUnit)];
19+
// equal value and unit
20+
yield return [Mass.FromGrams(42), Mass.FromGrams(42)];
21+
// zero in another unit
22+
yield return [Mass.Zero, Mass.FromGrams(0)];
23+
// same quantity in another unit
24+
yield return [Mass.FromGrams(42), Mass.FromMilligrams(42000)];
25+
// same quantity in another unit (in reverse)
26+
yield return [Mass.FromMilligrams(42000), Mass.FromGrams(42)];
27+
// different value and same unit
28+
yield return [Mass.FromGrams(42), Mass.FromGrams(42.1)];
29+
// huge values, same unit
30+
yield return [Mass.FromGrams(-1e37), Mass.FromGrams(1 / 1e12)];
31+
// huge values, different units
32+
yield return [Mass.FromGrams(-1e37), Mass.FromMilligrams(1 / 1e12)];
33+
// Math.PI, same unit
34+
yield return [Mass.FromGrams(Math.PI), Mass.FromGrams(Math.PI)];
35+
// Math.PI, different units
36+
yield return [Mass.FromGrams(Math.PI), Mass.FromMilligrams(Math.PI)];
37+
// very close fractions, same units
38+
yield return [Mass.FromGrams(12.3456789987654321), Mass.FromGrams(12.3456789987654322)];
39+
}
40+
41+
[Benchmark]
42+
[ArgumentsSource(nameof(Operands))]
43+
public bool Equals(Mass a, Mass b)
44+
{
45+
return a.Equals(b);
46+
}
47+
48+
[Benchmark]
49+
[ArgumentsSource(nameof(Operands))]
50+
public bool EqualsTolerance(Mass a, Mass b)
51+
{
52+
return a.Equals(b, Tolerance);
53+
}
54+
55+
[Benchmark]
56+
[ArgumentsSource(nameof(Operands))]
57+
public bool GetHashCode(Mass a, Mass b)
58+
{
59+
return a.GetHashCode() == b.GetHashCode();
60+
}
61+
62+
[Benchmark]
63+
[ArgumentsSource(nameof(Operands))]
64+
public int CompareTo(Mass a, Mass b)
65+
{
66+
return a.CompareTo(b);
67+
}
68+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
using System;
2+
using BenchmarkDotNet.Attributes;
3+
using BenchmarkDotNet.Jobs;
4+
using UnitsNet.Units;
5+
6+
namespace UnitsNet.Benchmark.Conversions.FromString;
7+
8+
[MemoryDiagnoser]
9+
[SimpleJob(RuntimeMoniker.Net48)]
10+
[SimpleJob(RuntimeMoniker.Net80)]
11+
public class ParseUnitBenchmarks
12+
{
13+
private readonly Random _random = new(42);
14+
private string[] _densityUnits;
15+
private string[] _massUnits;
16+
private string[] _pressureUnits;
17+
private string[] _volumeFlowUnits;
18+
private string[] _volumeUnits = [];
19+
20+
[Params(1000)]
21+
public int NbAbbreviations { get; set; }
22+
23+
[GlobalSetup(Target = nameof(ParseMassUnit))]
24+
public void PrepareMassUnits()
25+
{
26+
_massUnits = _random.GetItems(["mg", "g", "kg", "lbs", "Mlbs"], NbAbbreviations);
27+
}
28+
29+
[GlobalSetup(Target = nameof(ParseVolumeUnit))]
30+
public void PrepareVolumeUnits()
31+
{
32+
_volumeUnits = _random.GetItems(["ml", "l", "L", "cm³", "m³"], NbAbbreviations);
33+
}
34+
35+
[GlobalSetup(Target = nameof(ParseDensityUnit))]
36+
public void PrepareDensityUnits()
37+
{
38+
_densityUnits = _random.GetRandomAbbreviations<DensityUnit>(UnitsNetSetup.Default.UnitAbbreviations, NbAbbreviations);
39+
}
40+
41+
[GlobalSetup(Target = nameof(ParsePressureUnit))]
42+
public void PreparePressureUnits()
43+
{
44+
_pressureUnits = _random.GetRandomAbbreviations<PressureUnit>(UnitsNetSetup.Default.UnitAbbreviations, NbAbbreviations);
45+
}
46+
47+
[GlobalSetup(Target = nameof(ParseVolumeFlowUnit))]
48+
public void PrepareVolumeFlowUnits()
49+
{
50+
_volumeFlowUnits = _random.GetRandomAbbreviations<VolumeFlowUnit>(UnitsNetSetup.Default.UnitAbbreviations, NbAbbreviations);
51+
}
52+
53+
[Benchmark(Baseline = true)]
54+
public MassUnit ParseMassUnit()
55+
{
56+
MassUnit unit = default;
57+
foreach (var unitToParse in _massUnits)
58+
{
59+
unit = Mass.ParseUnit(unitToParse);
60+
}
61+
62+
return unit;
63+
}
64+
65+
[Benchmark(Baseline = false)]
66+
public VolumeUnit ParseVolumeUnit()
67+
{
68+
VolumeUnit unit = default;
69+
foreach (var unitToParse in _volumeUnits)
70+
{
71+
unit = Volume.ParseUnit(unitToParse);
72+
}
73+
74+
return unit;
75+
}
76+
77+
[Benchmark(Baseline = false)]
78+
public DensityUnit ParseDensityUnit()
79+
{
80+
DensityUnit unit = default;
81+
foreach (var unitToParse in _densityUnits)
82+
{
83+
unit = Density.ParseUnit(unitToParse);
84+
}
85+
86+
return unit;
87+
}
88+
89+
[Benchmark(Baseline = false)]
90+
public PressureUnit ParsePressureUnit()
91+
{
92+
PressureUnit unit = default;
93+
foreach (var unitToParse in _pressureUnits)
94+
{
95+
unit = Pressure.ParseUnit(unitToParse);
96+
}
97+
98+
return unit;
99+
}
100+
101+
[Benchmark(Baseline = false)]
102+
public VolumeFlowUnit ParseVolumeFlowUnit()
103+
{
104+
VolumeFlowUnit unit = default;
105+
foreach (var unitToParse in _volumeFlowUnits)
106+
{
107+
unit = VolumeFlow.ParseUnit(unitToParse);
108+
}
109+
110+
return unit;
111+
}
112+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
// Licensed under MIT No Attribution, see LICENSE file at the root.
2+
// Copyright 2013 Andreas Gullberg Larsen ([email protected]). Maintained at https://github.com/angularsen/UnitsNet.
3+
4+
using System;
5+
using System.Linq;
6+
using System.Text;
7+
using BenchmarkDotNet.Attributes;
8+
using BenchmarkDotNet.Jobs;
9+
using UnitsNet.Units;
10+
11+
namespace UnitsNet.Benchmark.Conversions.FromString;
12+
13+
[MemoryDiagnoser]
14+
[SimpleJob(RuntimeMoniker.Net48)]
15+
[SimpleJob(RuntimeMoniker.Net80)]
16+
public class TryParseInvalidUnitBenchmarks
17+
{
18+
private readonly Random _random = new(42);
19+
private string[] _invalidUnits = [];
20+
21+
[Params(1000)]
22+
public int NbAbbreviations { get; set; }
23+
24+
[GlobalSetup]
25+
public void Setup()
26+
{
27+
_invalidUnits = Enumerable.Range(0, NbAbbreviations).Select(_ => GenerateInvalidUnit()).ToArray();
28+
}
29+
30+
private string GenerateInvalidUnit()
31+
{
32+
var sb = new StringBuilder();
33+
var length = _random.Next(1, 10);
34+
for (var i = 0; i < length; i++)
35+
{
36+
sb.Append((char)_random.Next('a', 'z'));
37+
}
38+
39+
return sb.ToString();
40+
}
41+
42+
43+
[Benchmark(Baseline = true)]
44+
public bool TryParseMassUnit()
45+
{
46+
var success = true;
47+
foreach (var unitToParse in _invalidUnits)
48+
{
49+
success = Mass.TryParseUnit(unitToParse, out MassUnit _);
50+
}
51+
52+
return success;
53+
}
54+
55+
[Benchmark(Baseline = false)]
56+
public bool TryParseVolumeUnit()
57+
{
58+
var success = true;
59+
foreach (var unitToParse in _invalidUnits)
60+
{
61+
success = Volume.TryParseUnit(unitToParse, out _);
62+
}
63+
64+
return success;
65+
}
66+
67+
[Benchmark(Baseline = false)]
68+
public bool ParseDensityUnit()
69+
{
70+
var success = true;
71+
foreach (var unitToParse in _invalidUnits)
72+
{
73+
success = Density.TryParseUnit(unitToParse, out _);
74+
}
75+
76+
return success;
77+
}
78+
79+
[Benchmark(Baseline = false)]
80+
public bool ParsePressureUnit()
81+
{
82+
var success = true;
83+
foreach (var unitToParse in _invalidUnits)
84+
{
85+
success = Pressure.TryParseUnit(unitToParse, out _);
86+
}
87+
88+
return success;
89+
}
90+
91+
[Benchmark(Baseline = false)]
92+
public bool ParseVolumeFlowUnit()
93+
{
94+
var success = true;
95+
foreach (var unitToParse in _invalidUnits)
96+
{
97+
success = VolumeFlow.TryParseUnit(unitToParse, out _);
98+
}
99+
100+
return success;
101+
}
102+
}

0 commit comments

Comments
 (0)