为了账号安全,请及时绑定邮箱和手机立即绑定

子数组上的 C# LINQ OrderBy

子数组上的 C# LINQ OrderBy

C#
斯蒂芬大帝 2022-11-21 16:10:03
使用 C# LINQ,我们是否可以仅对 List 的一系列元素进行 OrderBy 而保留其他元素?例如,输入列表是 {"a","b","c","d","e"},想象中的 OrderByDescending 是这样的:OrderByDescending(delegate d,int start_index,int end_index) l=l.OrderByDescending(x=>x,1,3).ToList();结果是:{"a","d","c","b","e"}没有这个功能,我需要拆分/LINQ Orderby/rejoin,这会失去 LINQ 的精神。
查看完整描述

2 回答

?
婷婷同学_

TA贡献1844条经验 获得超8个赞

您可以使用Skip并Take实现它。


var input = new[] { "a", "b", "c", "d", "e" };


var res = input.Take(1)

               .Concat(input.Skip(1).Take(3).OrderByDescending(e => e))

               .Concat(input.Skip(4));

你也可以像这样做一个扩展方法


public static class IEnumerableExt

{

    public static IEnumerable<TSource> OrderRangeByDescending<TSource, TKey>(this IEnumerable<TSource> input, Func<TSource, TKey> keySelector, int from, int length)

    {

        return input.Take(from)

                    .Concat(input.Skip(from).Take(length).OrderByDescending(keySelector))

                    .Concat(input.Skip(from + length));

    }


    public static IEnumerable<TSource> OrderRangeBy<TSource, TKey>(this IEnumerable<TSource> input, Func<TSource, TKey> keySelector, int from, int length)

    {

        return input.Take(from)

                    .Concat(input.Skip(from).Take(length).OrderBy(keySelector))

                    .Concat(input.Skip(from + length));

    }

}

var input = new[] { "a", "b", "c", "d", "e" };


var res = input.OrderRangeByDescending(e => e, 1, 3);


查看完整回答
反对 回复 2022-11-21
?
呼唤远方

TA贡献1856条经验 获得超11个赞

重复调用Skip和Take可能会对性能产生影响,特别是如果源是由密集计算生成的。最佳解决方案将要求对源的读取仅发生一次。这可以通过将源拆分为多个枚举但使用单个枚举器来实现。拥有这样一个 Splitter 将使我们能够很容易地实现OrderRangeBy/OrderRangeByDescending方法:


public static IEnumerable<TSource> OrderRangeBy<TSource, TKey>(

    this IEnumerable<TSource> source, Func<TSource, TKey> keySelector,

    int startIndex, int endIndexExclusive)

{

    var parts = source.Split(startIndex, endIndexExclusive);

    return parts[0].Concat(parts[1].OrderBy(keySelector)).Concat(parts[2]);

}


public static IEnumerable<TSource> OrderRangeByDescending<TSource, TKey>(

    this IEnumerable<TSource> source, Func<TSource, TKey> keySelector,

    int startIndex, int endIndexExclusive)

{

    var parts = source.Split(startIndex, endIndexExclusive);

    return parts[0].Concat(parts[1].OrderByDescending(keySelector)).Concat(parts[2]);

}

下面是 Splitter 的一个实现:


public static IEnumerable<TSource>[] Split<TSource>(

    this IEnumerable<TSource> source, params int[] indices)

{

    var parts = new IEnumerable<TSource>[indices.Length + 1];

    var enumerator = source.GetEnumerator();

    int index = 0;

    for (int i = 0; i < indices.Length; i++)

    {

        parts[i] = GetPart(indices[i]);

    }

    parts[indices.Length] = GetPart(Int32.MaxValue);

    return parts;

    IEnumerable<TSource> GetPart(int maxIndexExclusive)

    {

        if (index >= maxIndexExclusive) goto finish;

        while (enumerator.MoveNext())

        {

            yield return enumerator.Current;

            index++;

            if (index >= maxIndexExclusive) break;

        }

        finish: if (maxIndexExclusive == Int32.MaxValue) enumerator.Dispose();

    }

}


查看完整回答
反对 回复 2022-11-21
  • 2 回答
  • 0 关注
  • 119 浏览

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号