使用此数组int[]{ 1, 2, 3, 4, 7, 8, 11, 15,16,17,18 };
如何转换为此字符串数组"1-4","7-8","11","15-18"
建议?Linq?
dtb*_*dtb 43
var array = new int[] { 1, 2, 3, 4, 7, 8, 11, 15, 16, 17, 18 };
var result = string.Join(",", array
.Distinct()
.OrderBy(x => x)
.GroupAdjacentBy((x, y) => x + 1 == y)
.Select(g => new int[] { g.First(), g.Last() }.Distinct())
.Select(g => string.Join("-", g)));
Run Code Online (Sandbox Code Playgroud)
同
public static class LinqExtensions
{
public static IEnumerable<IEnumerable<T>> GroupAdjacentBy<T>(
this IEnumerable<T> source, Func<T, T, bool> predicate)
{
using (var e = source.GetEnumerator())
{
if (e.MoveNext())
{
var list = new List<T> { e.Current };
var pred = e.Current;
while (e.MoveNext())
{
if (predicate(pred, e.Current))
{
list.Add(e.Current);
}
else
{
yield return list;
list = new List<T> { e.Current };
}
pred = e.Current;
}
yield return list;
}
}
}
}
Run Code Online (Sandbox Code Playgroud)
你不需要 Linq;事实上,最简单的解决方案需要知道数组中的三个位置(您的起始编号、当前编号和当前编号之后的下一个编号),而 Enumerables 并不适合这些位置。
尝试这个:
var start = 0;
var end = 0;
var write = false;
var builder = new StringBuilder();
for(var i=0; i<array.Length; i++)
{
//arranged this way to avoid ArrayOutOfBoundException
//if the next index doesn't exist or isn't one greater than the current,
//the current index is the end of our incremental range.
if(i+1 == array.Length || array[i+1] > array[i] + 1)
{
end = i;
write = true;
}
if(write)
{
if(end - start == 0) //one number
builder.Append(String.Format("{0}, ", array[start]);
else //multi-number range
builder.Append(String.Format("{0}-{1}, ", array[start], array[end]);
start = i+1;
end = i+1; //not really necessary but avoids any possible case of counting backwards
write = false;
}
}
Run Code Online (Sandbox Code Playgroud)
您可以continue在循环逻辑的早期重新安排它以减少代码嵌套,并删除一些变量;您将获得几毫秒的执行时间。在取出 String 之前,您还需要修剪掉 StringBuilder 末尾的最后两个字符(尾随逗号和空格)。
| 归档时间: |
|
| 查看次数: |
6343 次 |
| 最近记录: |