forked from ddrilling/AsbCloudServer
Add BinarySearch extension for Span<T>
This commit is contained in:
parent
1dee254427
commit
08e3e34162
@ -37,5 +37,44 @@ namespace AsbCloudInfrastructure
|
|||||||
var dateTz = date.ToOffset(TimeSpan.FromHours(remoteTimezoneOffsetHours));
|
var dateTz = date.ToOffset(TimeSpan.FromHours(remoteTimezoneOffsetHours));
|
||||||
return dateTz.DateTime;
|
return dateTz.DateTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Поиск индекса близкого к value значения в span.
|
||||||
|
/// Элементы в span должны быть отсортированы по возрастанию по полю propertyGetter.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T"></typeparam>
|
||||||
|
/// <typeparam name="TProp"></typeparam>
|
||||||
|
/// <param name="span"></param>
|
||||||
|
/// <param name="propertyGetter">метод получения свойства из объекта класса T</param>
|
||||||
|
/// <param name="value">искомое значение</param>
|
||||||
|
/// <param name="maxIterations">максимальное кол-во итераций. Прим.:
|
||||||
|
/// в span из 4000 записей достаточно 9-ти итераций
|
||||||
|
/// 8_000 => 10 итераций;
|
||||||
|
/// 16_000 => 11;
|
||||||
|
/// ...
|
||||||
|
/// 256_000 => 15;
|
||||||
|
/// При недостаточном кол-ве итераций индекс может оказаться близко, но это будет не ближайшее значение
|
||||||
|
/// </param>
|
||||||
|
/// <returns></returns>
|
||||||
|
public static int BinarySearch<T, TProp>(this Span<T> span, Func<T, TProp> propertyGetter, TProp value, int maxIterations = 16)
|
||||||
|
where TProp : IComparable
|
||||||
|
{
|
||||||
|
if (span.Length < 2 || --maxIterations < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
var indexOfMiddle = span.Length >> 1;
|
||||||
|
|
||||||
|
var comparison = value.CompareTo(propertyGetter(span[indexOfMiddle]));
|
||||||
|
|
||||||
|
if (comparison > 0)
|
||||||
|
{
|
||||||
|
var index = indexOfMiddle + 1 + BinarySearch(span[(indexOfMiddle + 1)..], propertyGetter, value, maxIterations);
|
||||||
|
return index < span.Length ? index : span.Length - 1;
|
||||||
|
}
|
||||||
|
else if (comparison < 0)
|
||||||
|
return BinarySearch(span[..(indexOfMiddle)], propertyGetter, value, maxIterations);
|
||||||
|
else //if (comparison == 0)
|
||||||
|
return indexOfMiddle;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user