123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691 |
- #nullable enable
- using System;
- using System.Collections.Concurrent;
- using System.Collections.Generic;
- using System.Diagnostics;
- using System.Linq;
- using System.Linq.Expressions;
- using System.Threading;
- using System.Threading.Tasks;
- namespace Masuit.Tools
- {
- public static partial class IEnumerableExtensions
- {
- #if NET6_0
- #else
- /// <summary>
- /// 按字段去重
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TKey"></typeparam>
- /// <param name="source"></param>
- /// <param name="keySelector"></param>
- /// <returns></returns>
- public static IEnumerable<TSource> DistinctBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
- {
- var hash = new HashSet<TKey>();
- return source.Where(p => hash.Add(keySelector(p)));
- }
- #endif
- /// <summary>
- /// 添加多个元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="values"></param>
- public static void AddRange<T>(this ICollection<T> @this, params T[] values)
- {
- foreach (var obj in values)
- {
- @this.Add(obj);
- }
- }
- /// <summary>
- /// 添加多个元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="values"></param>
- public static void AddRange<T>(this ConcurrentBag<T> @this, params T[] values)
- {
- foreach (var obj in values)
- {
- @this.Add(obj);
- }
- }
- /// <summary>
- /// 添加多个元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="values"></param>
- public static void AddRange<T>(this ConcurrentQueue<T> @this, params T[] values)
- {
- foreach (var obj in values)
- {
- @this.Enqueue(obj);
- }
- }
- /// <summary>
- /// 添加符合条件的多个元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="predicate"></param>
- /// <param name="values"></param>
- public static void AddRangeIf<T>(this ICollection<T> @this, Func<T, bool> predicate, params T[] values)
- {
- foreach (var obj in values)
- {
- if (predicate(obj))
- {
- @this.Add(obj);
- }
- }
- }
- /// <summary>
- /// 添加符合条件的多个元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="predicate"></param>
- /// <param name="values"></param>
- public static void AddRangeIf<T>(this ConcurrentBag<T> @this, Func<T, bool> predicate, params T[] values)
- {
- foreach (var obj in values)
- {
- if (predicate(obj))
- {
- @this.Add(obj);
- }
- }
- }
- /// <summary>
- /// 添加符合条件的多个元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="predicate"></param>
- /// <param name="values"></param>
- public static void AddRangeIf<T>(this ConcurrentQueue<T> @this, Func<T, bool> predicate, params T[] values)
- {
- foreach (var obj in values)
- {
- if (predicate(obj))
- {
- @this.Enqueue(obj);
- }
- }
- }
- /// <summary>
- /// 添加不重复的元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="values"></param>
- public static void AddRangeIfNotContains<T>(this ICollection<T> @this, params T[] values)
- {
- foreach (T obj in values)
- {
- if ([email protected](obj))
- {
- @this.Add(obj);
- }
- }
- }
- /// <summary>
- /// 移除符合条件的元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="this"></param>
- /// <param name="where"></param>
- public static void RemoveWhere<T>(this ICollection<T> @this, Func<T, bool> @where)
- {
- foreach (var obj in @this.Where(where).ToList())
- {
- @this.Remove(obj);
- }
- }
- /// <summary>
- /// 在元素之后添加元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="list"></param>
- /// <param name="condition">条件</param>
- /// <param name="value">值</param>
- public static void InsertAfter<T>(this IList<T> list, Func<T, bool> condition, T value)
- {
- foreach (var item in list.Select((item, index) => new { item, index }).Where(p => condition(p.item)).OrderByDescending(p => p.index))
- {
- if (item.index + 1 == list.Count)
- {
- list.Add(value);
- }
- else
- {
- list.Insert(item.index + 1, value);
- }
- }
- }
- /// <summary>
- /// 在元素之后添加元素
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="list"></param>
- /// <param name="index">索引位置</param>
- /// <param name="value">值</param>
- public static void InsertAfter<T>(this IList<T> list, int index, T value)
- {
- foreach (var item in list.Select((v, i) => new { Value = v, Index = i }).Where(p => p.Index == index).OrderByDescending(p => p.Index))
- {
- if (item.Index + 1 == list.Count)
- {
- list.Add(value);
- }
- else
- {
- list.Insert(item.Index + 1, value);
- }
- }
- }
- /// <summary>
- /// 转HashSet
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static HashSet<TResult> ToHashSet<T, TResult>(this IEnumerable<T> source, Func<T, TResult> selector)
- {
- var set = new HashSet<TResult>();
- set.UnionWith(source.Select(selector));
- return set;
- }
- /// <summary>
- /// 遍历IEnumerable
- /// </summary>
- /// <param name="objs"></param>
- /// <param name="action">回调方法</param>
- /// <typeparam name="T"></typeparam>
- public static void ForEach<T>(this IEnumerable<T> objs, Action<T> action)
- {
- foreach (var o in objs)
- {
- action(o);
- }
- }
- /// <summary>
- /// 异步foreach
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="source"></param>
- /// <param name="maxParallelCount">最大并行数</param>
- /// <param name="action"></param>
- /// <param name="cancellationToken"></param>
- /// <returns></returns>
- public static async Task ForeachAsync<T>(this IEnumerable<T> source, Func<T, Task> action, int maxParallelCount, CancellationToken cancellationToken = default)
- {
- if (Debugger.IsAttached)
- {
- foreach (var item in source)
- {
- await action(item);
- }
- return;
- }
- var list = new List<Task>();
- foreach (var item in source)
- {
- if (cancellationToken.IsCancellationRequested)
- {
- return;
- }
- list.Add(action(item));
- if (list.Count >= maxParallelCount)
- {
- await Task.WhenAll(list);
- list.Clear();
- }
- }
- await Task.WhenAll(list);
- }
- /// <summary>
- /// 异步foreach
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="source"></param>
- /// <param name="action"></param>
- /// <returns></returns>
- public static Task ForeachAsync<T>(this IEnumerable<T> source, Func<T, Task> action, CancellationToken cancellationToken = default)
- {
- return ForeachAsync(source, action, source.Count(), cancellationToken);
- }
- /// <summary>
- /// 异步Select
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static Task<TResult[]> SelectAsync<T, TResult>(this IEnumerable<T> source, Func<T, Task<TResult>> selector)
- {
- return Task.WhenAll(source.Select(selector));
- }
- /// <summary>
- /// 异步Select
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static Task<TResult[]> SelectAsync<T, TResult>(this IEnumerable<T> source, Func<T, int, Task<TResult>> selector)
- {
- return Task.WhenAll(source.Select(selector));
- }
- /// <summary>
- /// 异步For
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <param name="maxParallelCount">最大并行数</param>
- /// <param name="cancellationToken">取消口令</param>
- /// <returns></returns>
- public static async Task ForAsync<T>(this IEnumerable<T> source, Func<T, int, Task> selector, int maxParallelCount, CancellationToken cancellationToken = default)
- {
- int index = 0;
- if (Debugger.IsAttached)
- {
- foreach (var item in source)
- {
- await selector(item, index);
- index++;
- }
- return;
- }
- var list = new List<Task>();
- foreach (var item in source)
- {
- if (cancellationToken.IsCancellationRequested)
- {
- return;
- }
- list.Add(selector(item, index));
- Interlocked.Add(ref index, 1);
- if (list.Count >= maxParallelCount)
- {
- await Task.WhenAll(list);
- list.Clear();
- }
- }
- await Task.WhenAll(list);
- }
- /// <summary>
- /// 异步For
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <param name="cancellationToken">取消口令</param>
- /// <returns></returns>
- public static Task ForAsync<T>(this IEnumerable<T> source, Func<T, int, Task> selector, CancellationToken cancellationToken = default)
- {
- return ForAsync(source, selector, source.Count(), cancellationToken);
- }
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static TResult MaxOrDefault<TSource, TResult>(this IQueryable<TSource> source, Expression<Func<TSource, TResult>> selector) => source.Select(selector).DefaultIfEmpty().Max();
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TResult MaxOrDefault<TSource, TResult>(this IQueryable<TSource> source, Expression<Func<TSource, TResult>> selector, TResult defaultValue) => source.Select(selector).DefaultIfEmpty(defaultValue).Max();
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <returns></returns>
- public static TSource MaxOrDefault<TSource>(this IQueryable<TSource> source) => source.DefaultIfEmpty().Max();
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TSource MaxOrDefault<TSource>(this IQueryable<TSource> source, TSource defaultValue) => source.DefaultIfEmpty(defaultValue).Max();
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TResult MaxOrDefault<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> selector, TResult defaultValue) => source.Select(selector).DefaultIfEmpty(defaultValue).Max();
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static TResult MaxOrDefault<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> selector) => source.Select(selector).DefaultIfEmpty().Max();
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <returns></returns>
- public static TSource MaxOrDefault<TSource>(this IEnumerable<TSource> source) => source.DefaultIfEmpty().Max();
- /// <summary>
- /// 取最大值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TSource MaxOrDefault<TSource>(this IEnumerable<TSource> source, TSource defaultValue) => source.DefaultIfEmpty(defaultValue).Max();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static TResult MinOrDefault<TSource, TResult>(this IQueryable<TSource> source, Expression<Func<TSource, TResult>> selector) => source.Select(selector).DefaultIfEmpty().Min();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TResult MinOrDefault<TSource, TResult>(this IQueryable<TSource> source, Expression<Func<TSource, TResult>> selector, TResult defaultValue) => source.Select(selector).DefaultIfEmpty(defaultValue).Min();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <returns></returns>
- public static TSource MinOrDefault<TSource>(this IQueryable<TSource> source) => source.DefaultIfEmpty().Min();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TSource MinOrDefault<TSource>(this IQueryable<TSource> source, TSource defaultValue) => source.DefaultIfEmpty(defaultValue).Min();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static TResult MinOrDefault<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> selector) => source.Select(selector).DefaultIfEmpty().Min();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <typeparam name="TResult"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TResult MinOrDefault<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> selector, TResult defaultValue) => source.Select(selector).DefaultIfEmpty(defaultValue).Min();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <returns></returns>
- public static TSource MinOrDefault<TSource>(this IEnumerable<TSource> source) => source.DefaultIfEmpty().Min();
- /// <summary>
- /// 取最小值
- /// </summary>
- /// <typeparam name="TSource"></typeparam>
- /// <param name="source"></param>
- /// <param name="defaultValue"></param>
- /// <returns></returns>
- public static TSource MinOrDefault<TSource>(this IEnumerable<TSource> source, TSource defaultValue) => source.DefaultIfEmpty(defaultValue).Min();
- /// <summary>
- /// 标准差
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="source"></param>
- /// <param name="selector"></param>
- /// <returns></returns>
- public static TResult StandardDeviation<T, TResult>(this IEnumerable<T> source, Func<T, TResult> selector) where TResult : IConvertible
- {
- return StandardDeviation(source.Select(t => selector(t).ConvertTo<double>())).ConvertTo<TResult>();
- }
- /// <summary>
- /// 标准差
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="source"></param>
- /// <returns></returns>
- public static T StandardDeviation<T>(this IEnumerable<T> source) where T : IConvertible
- {
- return StandardDeviation(source.Select(t => t.ConvertTo<double>())).ConvertTo<T>();
- }
- /// <summary>
- /// 标准差
- /// </summary>
- /// <param name="source"></param>
- /// <returns></returns>
- public static double StandardDeviation(this IEnumerable<double> source)
- {
- double result = 0;
- int count = source.Count();
- if (count > 1)
- {
- double avg = source.Average();
- double sum = source.Sum(d => (d - avg) * (d - avg));
- result = Math.Sqrt(sum / count);
- }
- return result;
- }
- /// <summary>
- /// 随机排序
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="source"></param>
- /// <returns></returns>
- public static IOrderedEnumerable<T> OrderByRandom<T>(this IEnumerable<T> source)
- {
- return source.OrderBy(_ => Guid.NewGuid());
- }
- /// <summary>
- /// 序列相等
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="first"></param>
- /// <param name="second"></param>
- /// <param name="condition"></param>
- /// <returns></returns>
- public static bool SequenceEqual<T>(this IEnumerable<T> first, IEnumerable<T> second, Func<T, T, bool> condition)
- {
- if (first is ICollection<T> source1 && second is ICollection<T> source2)
- {
- if (source1.Count != source2.Count)
- {
- return false;
- }
- if (source1 is IList<T> list1 && source2 is IList<T> list2)
- {
- int count = source1.Count;
- for (int index = 0; index < count; ++index)
- {
- if (!condition(list1[index], list2[index]))
- {
- return false;
- }
- }
- return true;
- }
- }
- using IEnumerator<T> enumerator1 = first.GetEnumerator();
- using IEnumerator<T> enumerator2 = second.GetEnumerator();
- while (enumerator1.MoveNext())
- {
- if (!enumerator2.MoveNext() || !condition(enumerator1.Current, enumerator2.Current))
- {
- return false;
- }
- }
- return !enumerator2.MoveNext();
- }
- /// <summary>
- /// 序列相等
- /// </summary>
- /// <typeparam name="T1"></typeparam>
- /// <typeparam name="T2"></typeparam>
- /// <param name="first"></param>
- /// <param name="second"></param>
- /// <param name="condition"></param>
- /// <returns></returns>
- public static bool SequenceEqual<T1, T2>(this IEnumerable<T1> first, IEnumerable<T2> second, Func<T1, T2, bool> condition)
- {
- if (first is ICollection<T1> source1 && second is ICollection<T2> source2)
- {
- if (source1.Count != source2.Count)
- {
- return false;
- }
- if (source1 is IList<T1> list1 && source2 is IList<T2> list2)
- {
- int count = source1.Count;
- for (int index = 0; index < count; ++index)
- {
- if (!condition(list1[index], list2[index]))
- {
- return false;
- }
- }
- return true;
- }
- }
- using IEnumerator<T1> enumerator1 = first.GetEnumerator();
- using IEnumerator<T2> enumerator2 = second.GetEnumerator();
- while (enumerator1.MoveNext())
- {
- if (!enumerator2.MoveNext() || !condition(enumerator1.Current, enumerator2.Current))
- {
- return false;
- }
- }
- return !enumerator2.MoveNext();
- }
- /// <summary>
- /// 对比两个集合哪些是新增的、删除的、修改的
- /// </summary>
- /// <typeparam name="T1"></typeparam>
- /// <typeparam name="T2"></typeparam>
- /// <param name="olds"></param>
- /// <param name="news"></param>
- /// <param name="key1Selector">对比因素属性</param>
- /// <param name="key2Selector">对比因素属性</param>
- /// <returns></returns>
- public static (List<T2> adds, List<T1> remove, List<T1> updates) CompareChanges<T1, T2>(this IEnumerable<T1> olds, IEnumerable<T2> news, Func<T1, object> key1Selector, Func<T2, object> key2Selector)
- {
- return (news.Where(c => olds.All(m => key1Selector(m) != key2Selector(c))).ToList(), olds.Where(m => news.All(c => key2Selector(c) != key1Selector(m))).ToList(), olds.Where(m => news.Any(c => key1Selector(m) == key2Selector(c))).ToList());
- }
- /// <summary>
- /// 对比两个集合哪些是新增的、删除的、修改的
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="olds"></param>
- /// <param name="news"></param>
- /// <param name="keySelector">对比因素属性</param>
- /// <returns></returns>
- public static (List<T> adds, List<T> remove, List<T> updates) CompareChanges<T>(this IEnumerable<T> olds, IEnumerable<T> news, Func<T, object> keySelector)
- {
- return (news.Where(c => olds.All(m => keySelector(m) != keySelector(c))).ToList(), olds.Where(m => news.All(c => keySelector(c) != keySelector(m))).ToList(), olds.Where(m => news.Any(c => keySelector(m) == keySelector(c))).ToList());
- }
- }
- }
|