Преобразование наборов целых чисел в диапазоны с использованием C # - PullRequest
7 голосов
/ 10 ноября 2010

Какой самый идиоматический способ преобразования набора целых чисел в набор диапазонов?

Например, заданный набор {0, 1, 2, 3, 4, 7, 8, 9, 11} Iхочу получить {{0,4}, {7,9}, {11,11}} с использованием C #

На этот вопрос уже дан ответ в C ++ @ Решение в C ++

Ответы [ 3 ]

11 голосов
/ 10 ноября 2010

Это не очень эффективно, но - это идиоматика:

var nums = new HashSet<int>{0, 1, 2, 3, 4, 7, 8, 9, 11};
IEnumerable<Tuple<int, int>> ranges = Enumerable.Zip(
    nums.Where(n => !nums.Contains(n - 1)),
    nums.Where(n => !nums.Contains(n + 1)),
    Tuple.Create);

Более эффективно, при условии, что оно отсортировано:

public IEnumerable<Tuple<int, int>> GetContiguousRanges(IEnumerable<int> nums)
{
    int start = nums.First();
    int last = start - 1;
    foreach (int i in nums)
    {
        if (i != last + 1)
        {
            yield return Tuple.Create(start, last);
            start = i;
        }
        last = i;
    }
    yield return Tuple.Create(start, last);
}
3 голосов
/ 10 ноября 2010

Это должна быть довольно простая транслитерация из поста, который вы упомянули.Убедитесь, что вы поместили этот код в класс где-то, код C # должен быть в классе.Я предполагаю, что вы не очень знакомы с C #, поэтому я сделаю достаточно, чтобы показать сходства и различия, и, надеюсь, вы справитесь с остальными.

struct Range
{
    public Range (int start, int end) { this.start = start; this.end = end; }
    public int start;
    public int end;
}

public static void SetToRanges(Dictionary<int,bool> indices, List<Range> ranges) 
{
    Range r = new Range(int.MinValue, int.MaxValue);
    foreach (int i in indices.Keys)
    {
        // translate rest of code here
    }
    ranges.Add(r);
    return ranges;
}

Для более идиоматического решения я бывернуть IEnumerable<Range>, поэтому «список» может быть построен и повторен одновременно:

public static IEnumerable<Range> SetToRanges(Dictionary<int, bool> indices)
{
     // instead of "ranges.Add(r)", use "yield return r".
     // This returns multiple values in order from the function, that can
     // be iterated with "foreach (Range i in SetToRanges(foo))"
}
0 голосов
/ 10 ноября 2010

Попробуйте K-означает кластеризацию, чтобы получить диапазоны.Вам нужно будет указать, сколько разных диапазонов вы хотите.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...