Управление диапазонами с помощью LINQ challenge
Даны следующие цифры (обозначающие дни недели): 1,2,3,4,5,6,7
,
Вот несколько примеров комбинации и их желаемый результат:
1,2,3,5,6,7
->1-3,5-7
1,3,5,7
->1,3,5,7
1,2,5,6
->1,2,5,6
1,2,3,6,7
->1-3,6,7
Идея состоит в том, что 3 или более последовательных дня становятся диапазоном, в то время как отдельные или непоследующие дни отображаются отдельно (или лучше выбрать диапазон, начинающийся с 2).
Я не знаю, с чего начать, если я напишу сложный if
Функция ed или это можно сделать с помощью одной из функций LINQ?
Есть сочные предложения?
Я использовал числа, чтобы упростить идею диапазонов, но в моем коде у меня есть помеченное перечисление, объявленное следующим образом:
[Flags]
public enum DaysOfWeek
{
Sunday = 0x1,
Monday = 0x2,
Tuesday = 0x4,
Wednesday = 0x8,
Thursday = 0x10,
Friday = 0x20,
Saturday = 0x40
}
У меня есть сущность OpeningTimes
с полем DaysOfWeek
, который сообщает, к каким дням недели относятся часовые диапазоны (определенные в другом свойстве) этой сущности.
Таким образом, чтобы получить то, что я использую (чтобы действительно получить числа, я бы добавил Select
используя индекс + 1):
var days = Enum.GetValues(typeof(DaysOfWeek))
.Cast<DaysOfWeek>()
.Where(dow => Model.DaysOfWeek.HasFlag(dow));
Я думаю, что идея состоит в том, чтобы сначала удалить числа в диапазоне.
Я считаю, что я ищу функцию агрегации, которая также получает предыдущее значение и может возвращать другой тип значения, поэтому я могу создать функцию, которая, если текущее значение -1 равно prev. Значение, я жду следующего значения, пока диапазон не будет последовательным (или если элемент сам за себя), когда я возвращаю последний объем как анонимный объект и начинаю работать над новым.
Затем я сделаю функцию форматирования, которая говорит if (item.First != item.Last) string.Join("-", item.First, Item.Last);
3 ответа
Интересная проблема. Я решил для удобства чтения иметь класс, представляющий диапазон:
class NumberRange
{
public int Start { get; set;}
public int End { get; set;}
public override string ToString()
{
return Start == End ? Start.ToString() : String.Format("{0}-{1}",Start,End);
}
}
и метод расширения, чтобы превратить IEnumerable из упорядоченных целых в IEnumerable из диапазонов:
public static IEnumerable<NumberRange> ToRanges(this IEnumerable<int> numbers)
{
NumberRange currentRange = null;
foreach(var number in numbers)
{
if (currentRange == null)
currentRange = new NumberRange() { Start = number, End = number };
else if (number == currentRange.End + 1)
currentRange.End = number;
else
{
yield return currentRange;
currentRange = new NumberRange { Start = number, End = number };
}
}
if (currentRange != null)
{
yield return currentRange;
}
}
И с этим на месте вы можете получить диапазоны и отформатировать их так, как вы хотите:
String.Join(",",
new int[] { 1,2,3,5,7,8,9,11 }
.ToRanges()
.Select(r => r.ToString()))
Вот что я подумал:
void Main()
{
Console.WriteLine(AggregateString(new int[]{1,2,3,5,6,7})); //1-3,5-7
Console.WriteLine(AggregateString(new int[]{1,3,5,7})); //1,3,5,7
Console.WriteLine(AggregateString(new int[]{1,2,5,6})); //1,2,5,6
Console.WriteLine(AggregateString(new int[]{1,2,3,6,7 })); //1-3,6,7
}
string AggregateString(int[] ary)
{
List<List<int>> result=new List<List<int>>();
ary.Aggregate((m,n)=>
{
if(m == n-1)
{
if(result.LastOrDefault()!=null && result.LastOrDefault().Last() ==m)
result.Last().Add(n);
else
result.Add(new List<int>{m,n});
}
else
{
if(result.LastOrDefault()==null)
result.Add(new List<int>{m,n});
else result.Add(new List<int>{n});
}
return n;
});
return string.Join(",", result.Select(s=>s.Count()>2?
string.Join("-",new string[]{s.First().ToString(),s.Last().ToString()}) :
string.Join(",",s.Select(x=>x.ToString()).ToArray())).ToArray());
}
Вот мой взгляд на это. (К сожалению, я не мог предотвратить дублирование одного раздела:
static IEnumerable<string> GetRange(IEnumerable<int> range)
{
using(IEnumerator<int> iter = range.GetEnumerator())
if(iter.MoveNext())
{
int last = iter.Current;
int start = iter.Current;
while(iter.MoveNext())
{
int curr = iter.Current;
if (curr == last+1)
{
last = curr;
continue;
}
// found gap
if (start == last) // one isolated value
{
yield return start.ToString();
}
else if (last - start == 1) // two in a row.
{
yield return start.ToString();
yield return last.ToString();
}
else
{
yield return string.Format("{0}-{1}", start,last);
}
start = curr;
last = curr;
}
if (start == last) // one isolated value
{
yield return start.ToString();
}
else if (last - start == 1) // two in a row.
{
yield return start.ToString();
yield return last.ToString();
}
else
{
yield return string.Format("{0}-{1}", start,last);
}
}
}