-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathProcedureResults.cs
100 lines (83 loc) · 2.73 KB
/
ProcedureResults.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Diagnostics.Contracts;
using System.Linq;
namespace Open.Numeric;
public class ProcedureResults : IProcedureResult<ImmutableArray<double>>
{
public int Count { get; }
public ImmutableArray<double> Sum { get; }
public ImmutableArray<double> Average { get; }
public static ProcedureResults Empty { get; } = new();
protected ProcedureResults()
{
Count = 0;
Sum = ImmutableArray<double>.Empty;
Average = ImmutableArray<double>.Empty;
}
#if NET6_0_OR_GREATER
public ProcedureResults(ImmutableArray<double> sum, int count)
{
Sum = sum;
Count = count;
Average = sum.Select(v => count == 0 ? double.NaN : v / count).ToImmutableArray();
}
#else
public ProcedureResults(in ImmutableArray<double> sum, int count)
{
Sum = sum;
Count = count;
Average = sum.Select(v => count == 0 ? double.NaN : v / count).ToImmutableArray();
}
#endif
public ProcedureResults(IEnumerable<double> sum, int count)
: this(sum is ImmutableArray<double> s ? s : sum.ToImmutableArray(), count)
{
}
public ProcedureResults(ReadOnlySpan<double> sum, int count)
: this(ToImmutableArray(sum), count)
{
}
private static ImmutableArray<T> ToImmutableArray<T>(ReadOnlySpan<T> source)
{
var len = source.Length;
var builder = ImmutableArray.CreateBuilder<T>(len);
for (var i = 0; i < len; i++)
builder.Add(source[i]);
return builder.MoveToImmutable();
}
private static ImmutableArray<double> SumValues(IReadOnlyList<double> a, IReadOnlyList<double> b)
{
var len = a.Count;
if (len != b.Count)
throw new ArgumentException("Length mismatch.");
var builder = ImmutableArray.CreateBuilder<double>(len);
for (var i = 0; i < len; i++)
builder.Add(a[i] + b[i]);
return builder.MoveToImmutable();
}
private static ImmutableArray<double> SumValues(IReadOnlyList<double> a, ReadOnlySpan<double> b)
{
var len = a.Count;
if (len != b.Length)
throw new ArgumentException("Length mismatch.");
var builder = ImmutableArray.CreateBuilder<double>(len);
for (var i = 0; i < len; i++)
builder.Add(a[i] + b[i]);
return builder.MoveToImmutable();
}
public ProcedureResults Add(IReadOnlyList<double> values, int count = 1)
=> new(SumValues(Sum, values ?? throw new ArgumentNullException(nameof(values))), Count + count);
public ProcedureResults Add(ReadOnlySpan<double> values, int count = 1)
=> new(SumValues(Sum, values), Count + count);
public static ProcedureResults operator +(ProcedureResults a, ProcedureResults b)
{
if (a is null)
throw new ArgumentNullException(nameof(a));
if (b is null)
throw new ArgumentNullException(nameof(b));
Contract.EndContractBlock();
return new(SumValues(a.Sum, b.Sum), a.Count + b.Count);
}
}