确定数字是否在指定的范围集内

Ste*_*nan 18 c# range

我正在寻找一种流畅的方法来确定一个数字是否在指定的范围内.我当前的代码看起来像这样:

int x = 500; // Could be any number

if ( ( x > 4199 && x < 6800 ) ||
     ( x > 6999 && x < 8200 ) ||
     ( x > 9999 && x < 10100 ) ||
     ( x > 10999 && x < 11100 ) ||
     ( x > 11999 && x < 12100 ) )
{
    // More awesome code
}
Run Code Online (Sandbox Code Playgroud)

有没有更好的方法呢?

Ant*_*lev 36

扩展方法?

bool Between(this int value, int left, int right)
{ 
   return value > left && value < right; 
}

if(x.Between(4199, 6800) || x.Between(6999, 8200) || ...)
Run Code Online (Sandbox Code Playgroud)

你也可以做这个糟糕的黑客:

bool Between(this int value, params int[] values)
{
    // Should be even number of items
    Debug.Assert(values.Length % 2 == 0); 

    for(int i = 0; i < values.Length; i += 2)
        if(!value.Between(values[i], values[i + 1])
            return false;

    return true;
}

if(x.Between(4199, 6800, 6999, 8200, ...)
Run Code Online (Sandbox Code Playgroud)

糟糕的黑客,改进:

class Range
{
    int Left { get; set; }
    int Right { get; set; }

    // Constructors, etc.
}

Range R(int left, int right)
{
    return new Range(left, right)
}

bool Between(this int value, params Range[] ranges)
{
    for(int i = 0; i < ranges.Length; ++i)
        if(value > ranges[i].Left && value < ranges[i].Right)
            return true;

    return false;
}

if(x.Between(R(4199, 6800), R(6999, 8200), ...))
Run Code Online (Sandbox Code Playgroud)

或者,更好(这不允许重复的下限):

bool Between(this int value, Dictionary<int, int> ranges)
{
    // Basically iterate over Key-Value pairs and check if value falls within that range
}

if(x.Between({ { 4199, 6800 }, { 6999, 8200 }, ... }
Run Code Online (Sandbox Code Playgroud)

  • 我肯定会*使范围不可变.变异性很糟糕...... (4认同)

Jon*_*eet 14

定义Range类型,然后创建一组范围和扩展方法,以查看值是否位于任何范围内.然后,您可以创建范围的集合,也可以创建一些单独的范围,而不是硬编码值,为它们提供有用的名称来解释您对它们感兴趣的原因:

static readonly Range InvalidUser = new Range(100, 200);
static readonly Range MilkTooHot = new Range (300, 400);

static readonly IEnumerable<Range> Errors =
    new List<Range> { InvalidUser, MilkTooHot };

...

// Normal LINQ (where Range defines a Contains method)
if (Errors.Any(range => range.Contains(statusCode))
// or (extension method on int)
if (statusCode.InAny(Errors))
// or (extension methods on IEnumerable<Range>)
if (Errors.Any(statusCode))
Run Code Online (Sandbox Code Playgroud)

您可能对Range作为MiscUtil一部分的泛型类型感兴趣.它允许以简单的方式迭代:

foreach (DateTime date in 19.June(1976).To(25.December(2005)).Step(1.Days()))
{
    // etc
}
Run Code Online (Sandbox Code Playgroud)

(显然,这也使用了一些与DateTime/TimeSpan相关的扩展方法,但是你明白了.)


Sco*_*vey 7

我个人更喜欢@Anton建议的扩展方法 - 但是如果你不能这样做,并且会坚持使用你当前的代码,我认为你可以通过颠倒每行上的第一组条件使其更具可读性,如下所示...

int x = 500; // Could be any number
if ( ( 4199 < x && x < 6800 ) ||
     ( 6999 < x && x < 8200 ) ||
     ( 9999 < x && x < 10100 ) ||
     ( 10999 < x && x < 11100 ) ||
     ( 11999 < x && x < 12100 ) )
{
    // More awesome code
}
Run Code Online (Sandbox Code Playgroud)