Poniższy kod obsługuje określanie granic w dowolnej kolejności (tj. bound1 <= bound2
Lub bound2 <= bound1
). Zauważyłem, że jest to przydatne do mocowania wartości obliczonych z równań liniowych ( y=mx+b
), w których nachylenie linii może rosnąć lub maleć.
Wiem: kod składa się z pięciu bardzo brzydkich operatorów wyrażeń warunkowych . Rzecz w tym, że działa , a poniższe testy to potwierdzają. Jeśli sobie tego życzysz, możesz dodać niepotrzebne nawiasy.
Możesz łatwo utworzyć inne przeciążenia dla innych typów liczbowych i po prostu skopiować / wkleić testy.
Ostrzeżenie: porównywanie liczb zmiennoprzecinkowych nie jest proste. Ten kod nie implementuje double
solidnie porównań. Użyj biblioteki porównawczej zmiennoprzecinkowej, aby zastąpić użycie operatorów porównania.
public static class MathExtensions
{
public static double Clamp(this double value, double bound1, double bound2)
{
return bound1 <= bound2 ? value <= bound1 ? bound1 : value >= bound2 ? bound2 : value : value <= bound2 ? bound2 : value >= bound1 ? bound1 : value;
}
}
Testy xUnit / FluentAssertions:
public class MathExtensionsTests
{
[Theory]
[InlineData(0, 0, 0, 0)]
[InlineData(0, 0, 2, 0)]
[InlineData(-1, 0, 2, 0)]
[InlineData(1, 0, 2, 1)]
[InlineData(2, 0, 2, 2)]
[InlineData(3, 0, 2, 2)]
[InlineData(0, 2, 0, 0)]
[InlineData(-1, 2, 0, 0)]
[InlineData(1, 2, 0, 1)]
[InlineData(2, 2, 0, 2)]
[InlineData(3, 2, 0, 2)]
public void MustClamp(double value, double bound1, double bound2, double expectedValue)
{
value.Clamp(bound1, bound2).Should().Be(expectedValue);
}
}