试试 IEnumerable 的另外 6 个小例子
IEnumerable 接口是 C# 开发过程中非常重要的接口,对于其特性和用法的了解是十分必要的。本文将通过6个小例子,来熟悉一下其简单的用法。
<!-- more -->
阅读建议
- 在阅读本篇时,建议先阅读前篇《试试IEnumerable的10个小例子》,更加助于读者理解。
- 阅读并理解本篇需要花费5-10分钟左右的时间,而且其中包含一些实践建议。建议先收藏本文,闲时阅读并实践。
全是源码
以下便是这6个小例子,相应的说明均标记在注释中。
每个以 TXX 开头命名的均是一个示例。建议从上往下阅读。
using System;
using System.Collections.Generic;
using System.Linq;
using FluentAssertions;
using Xunit;
using Xunit.Abstractions; namespace Try_More_On_IEnumerable
{
public class EnumerableTests2
{
private readonly ITestOutputHelper _testOutputHelper; public EnumerableTests2(
ITestOutputHelper testOutputHelper)
{
_testOutputHelper = testOutputHelper;
} [Fact]
public void T11分组合并()
{
var array1 = new[] {, , , , };
var array2 = new[] {, , , , }; // 通过本地方法合并两个数组为一个数据
var result1 = ConcatArray(array1, array2).ToArray(); // 使用 Linq 中的 Concat 来合并两个 IEnumerable 对象
var result2 = array1.Concat(array2).ToArray(); // 使用 Linq 中的 SelectMany 将 “二维数据” 拉平合并为一个数组
var result3 = new[] {array1, array2}.SelectMany(x => x).ToArray(); /**
* 使用 Enumerable.Range 生成一个数组,这个数据的结果为
* 0,1,2,3,4,5,6,7,8,9
*/
var result = Enumerable.Range(, ).ToArray(); // 通过以上三种方式合并的结果时相同的
result1.Should().Equal(result);
result2.Should().Equal(result);
result3.Should().Equal(result); IEnumerable<T> ConcatArray<T>(IEnumerable<T> source1, IEnumerable<T> source2)
{
foreach (var item in source1)
{
yield return item;
} foreach (var item in source2)
{
yield return item;
}
}
} [Fact]
public void T12拉平三重循环()
{
/**
* 通过本地函数获取 0-999 共 1000 个数字。
* 在 GetSomeData 通过三重循环构造这些数据
* 值得注意的是 GetSomeData 隐藏了三重循环的细节
*/
var result1 = GetSomeData(, , )
.ToArray(); /**
* 与 GetSomeData 方法对比,将“遍历”和“处理”两个逻辑进行了分离。
* “遍历”指的是三重循环本身。
* “处理”指的是三重循环最内部的加法过程。
* 这里通过 Select 方法,将“处理”过程抽离了出来。
* 这其实和 “T03分离条件”中使用 Where 使用的是相同的思想。
*/
var result2 = GetSomeData2(, , )
.Select(tuple => tuple.i * + tuple.j * + tuple.k)
.ToArray(); // 生成一个 0-999 的数组。
var result = Enumerable.Range(, ).ToArray(); result1.Should().Equal(result);
result2.Should().Equal(result); IEnumerable<int> GetSomeData(int maxI, int maxJ, int maxK)
{
for (var i = ; i < maxI; i++)
{
for (var j = ; j < maxJ; j++)
{
for (var k = ; k < maxK; k++)
{
yield return i * + j * + k;
}
}
}
} IEnumerable<(int i, int j, int k)> GetSomeData2(int maxI, int maxJ, int maxK)
{
for (var i = ; i < maxI; i++)
{
for (var j = ; j < maxJ; j++)
{
for (var k = ; k < maxK; k++)
{
yield return (i, j, k);
}
}
}
}
} private class TreeNode
{
public TreeNode()
{
Children = Enumerable.Empty<TreeNode>();
} /// <summary>
/// 当前节点的值
/// </summary>
public int Value { get; set; } /// <summary>
/// 当前节点的子节点列表
/// </summary>
public IEnumerable<TreeNode> Children { get; set; }
} [Fact]
public void T13遍历树()
{
/**
* 树结构如下:
* └─0
* ├─1
* │ └─3
* └─2
*/
var tree = new TreeNode
{
Value = ,
Children = new[]
{
new TreeNode
{
Value = ,
Children = new[]
{
new TreeNode
{
Value =
},
}
},
new TreeNode
{
Value =
},
}
}; // 深度优先遍历的结果
var dftResult = new[] {, , , }; // 通过迭代器实现深度优先遍历
var dft = DFTByEnumerable(tree).ToArray();
dft.Should().Equal(dftResult); // 使用堆栈配合循环算法实现深度优先遍历
var dftList = DFTByStack(tree).ToArray();
dftList.Should().Equal(dftResult); // 递归算法实现深度优先遍历
var dftByRecursion = DFTByRecursion(tree).ToArray();
dftByRecursion.Should().Equal(dftResult); // 广度优先遍历的结果
var bdfResult = new[] {, , , }; /**
* 通过迭代器实现广度优先遍历
* 此处未提供“通过队列配合循环算法”和“递归算法”实现广度优先遍历的两种算法进行对比。读者可以自行尝试。
*/
var bft = BFT(tree).ToArray();
bft.Should().Equal(bdfResult); /**
* 迭代器深度优先遍历
* depth-first traversal
*/
IEnumerable<int> DFTByEnumerable(TreeNode root)
{
yield return root.Value;
foreach (var child in root.Children)
{
foreach (var item in DFTByEnumerable(child))
{
yield return item;
}
}
} // 使用堆栈配合循环算法实现深度优先遍历
IEnumerable<int> DFTByStack(TreeNode root)
{
var result = new List<int>();
var stack = new Stack<TreeNode>();
stack.Push(root);
while (stack.TryPop(out var node))
{
result.Add(node.Value);
foreach (var nodeChild in node.Children.Reverse())
{
stack.Push(nodeChild);
}
} return result;
} // 递归算法实现深度优先遍历
IEnumerable<int> DFTByRecursion(TreeNode root)
{
var list = new List<int> {root.Value};
foreach (var rootChild in root.Children)
{
list.AddRange(DFTByRecursion(rootChild));
} return list;
} // 通过迭代器实现广度优先遍历
IEnumerable<int> BFT(TreeNode root)
{
yield return root.Value; foreach (var bftChild in BFTChildren(root.Children))
{
yield return bftChild;
} IEnumerable<int> BFTChildren(IEnumerable<TreeNode> children)
{
var tempList = new List<TreeNode>();
foreach (var treeNode in children)
{
tempList.Add(treeNode);
yield return treeNode.Value;
} foreach (var bftChild in tempList.SelectMany(treeNode => BFTChildren(treeNode.Children)))
{
yield return bftChild;
}
}
}
} [Fact]
public void T14搜索树()
{
/**
* 此处所指的搜索树是指在遍历树的基础上增加终结遍历的条件。
* 因为一般构建搜索树是为了找到第一个满足条件的数据,因此与单纯的遍历存在不同。
* 树结构如下:
* └─0
* ├─1
* │ └─3
* └─5
* └─2
*/ var tree = new TreeNode
{
Value = ,
Children = new[]
{
new TreeNode
{
Value = ,
Children = new[]
{
new TreeNode
{
Value =
},
}
},
new TreeNode
{
Value = ,
Children = new[]
{
new TreeNode
{
Value =
},
}
},
}
}; /**
* 有了深度优先遍历算法的情况下,再增加一个条件判断,便可以实现深度优先的搜索
* 搜索树中第一个大于等于 3 并且是奇数的数字
*/
var result = DFS(tree, x => x >= && x % == ); /**
* 搜索到的结果是3。
* 特别提出,如果使用广度优先搜索,结果应该是5。
* 读者可以通过 T13遍历树 中的广度优先遍历算法配合 FirstOrDefault 中相同的条件实现。
* 建议读者尝试以上代码尝试一下。
*/
result.Should().Be(); int DFS(TreeNode root, Func<int, bool> predicate)
{
var re = DFTByEnumerable(root)
.FirstOrDefault(predicate);
return re;
} // 迭代器深度优先遍历
IEnumerable<int> DFTByEnumerable(TreeNode root)
{
yield return root.Value;
foreach (var child in root.Children)
{
foreach (var item in DFTByEnumerable(child))
{
yield return item;
}
}
}
} [Fact]
public void T15分页()
{
var arraySource = new[] {, , , , , , , , , }; // 使用迭代器进行分页,每 3 个一页
var enumerablePagedResult = PageByEnumerable(arraySource, ).ToArray(); // 结果一共 4 页
enumerablePagedResult.Should().HaveCount();
// 最后一页只有一个数字,为 9
enumerablePagedResult.Last().Should().Equal(); // 通过常规的 Skip 和 Take 来分页是最为常见的办法。结果应该与上面的分页结果一样
var result3 = NormalPage(arraySource, ).ToArray(); result3.Should().HaveCount();
result3.Last().Should().Equal(); IEnumerable<IEnumerable<int>> PageByEnumerable(IEnumerable<int> source, int pageSize)
{
var onePage = new LinkedList<int>();
foreach (var i in source)
{
onePage.AddLast(i);
if (onePage.Count != pageSize)
{
continue;
} yield return onePage;
onePage = new LinkedList<int>();
} // 最后一页如果数据不足一页,也应该返回该页
if (onePage.Count > )
{
yield return onePage;
}
} IEnumerable<IEnumerable<int>> NormalPage(IReadOnlyCollection<int> source, int pageSize)
{
var pageCount = Math.Ceiling(1.0 * source.Count / pageSize);
for (var i = ; i < pageCount; i++)
{
var offset = i * pageSize;
var onePage = source
.Skip(offset)
.Take(pageSize);
yield return onePage;
}
} /**
* 从写法逻辑上来看,显然 NormalPage 的写法更容易让大众接受
* PageByEnumerable 写法在仅仅只有在一些特殊的情况下才能体现性能上的优势,可读性上却不如 NormalPage
*/
} [Fact]
public void T16分页与多级缓存()
{
/**
* 获取 5 页数据,每页 2 个。
* 依次从 内存、Redis、ElasticSearch和数据库中获取数据。
* 先从内存中获取数据,如果内存中数据不足页,则从 Redis 中获取。
* 若 Redis 获取后还是不足页,进而从 ElasticSearch 中获取。依次类推,直到足页或者再无数据
*/
const int pageSize = ;
const int pageCount = ;
var emptyData = Enumerable.Empty<int>().ToArray(); /**
* 初始化各数据源的数据,除了内存有数据外,其他数据源均没有数据
*/
var memoryData = new[] {, , };
var redisData = emptyData;
var elasticSearchData = emptyData;
var databaseData = emptyData; var result = GetSourceData()
// ToPagination 是一个扩展方法。此处是为了体现链式调用的可读性,转而使用扩展方法,没有使用本地函数
.ToPagination(pageCount, pageSize)
.ToArray(); result.Should().HaveCount();
result[].Should().Equal(, );
result[].Should().Equal(); /**
* 初始化各数据源数据,各个数据源均有一些数据
*/
memoryData = new[] {, , };
redisData = new[] {, , };
elasticSearchData = new[] {, , };
databaseData = Enumerable.Range(, ).ToArray(); var result2 = GetSourceData()
.ToPagination(pageCount, pageSize)
.ToArray(); result2.Should().HaveCount();
result2[].Should().Equal(, );
result2[].Should().Equal(, );
result2[].Should().Equal(, );
result2[].Should().Equal(, );
result2[].Should().Equal(, ); IEnumerable<int> GetSourceData()
{
// 将多数据源的数据连接在一起
var data = GetDataSource()
.SelectMany(x => x);
return data; // 获取数据源
IEnumerable<IEnumerable<int>> GetDataSource()
{
// 将数据源依次返回
yield return GetFromMemory();
yield return GetFromRedis();
yield return GetFromElasticSearch();
yield return GetFromDatabase();
} IEnumerable<int> GetFromMemory()
{
_testOutputHelper.WriteLine("正在从内存中获取数据");
return memoryData;
} IEnumerable<int> GetFromRedis()
{
_testOutputHelper.WriteLine("正在从Redis中获取数据");
return redisData;
} IEnumerable<int> GetFromElasticSearch()
{
_testOutputHelper.WriteLine("正在从ElasticSearch中获取数据");
return elasticSearchData;
} IEnumerable<int> GetFromDatabase()
{
_testOutputHelper.WriteLine("正在从数据库中获取数据");
return databaseData;
}
} /**
* 值得注意的是:
* 由于 Enumerable 按需迭代的特性,如果将 result2 的所属页数改为只获取 1 页。
* 则在执行数据获取时,将不会再控制台中输出从 Redis、ElasticSearch和数据库中获取数据。
* 也就是说,并没有执行这些操作。读者可以自行修改以上代码,加深印象。
*/
}
} public static class EnumerableExtensions
{
/// <summary>
/// 将原数据分页
/// </summary>
/// <param name="source">数据源</param>
/// <param name="pageCount">页数</param>
/// <param name="pageSize">页大小</param>
/// <returns></returns>
public static IEnumerable<IEnumerable<int>> ToPagination(this IEnumerable<int> source,
int pageCount,
int pageSize)
{
var maxCount = pageCount * pageSize;
var countNow = ;
var onePage = new LinkedList<int>();
foreach (var i in source)
{
onePage.AddLast(i);
countNow++; // 如果获取的数量已经达到了分页所需要的总数,则停止进一步迭代
if (countNow == maxCount)
{
break;
} if (onePage.Count != pageSize)
{
continue;
} yield return onePage;
onePage = new LinkedList<int>();
} // 最后一页如果数据不足一页,也应该返回该页
if (onePage.Count > )
{
yield return onePage;
}
}
}
}
源码说明
以上示例的源代码放置于博客示例代码库中。
项目采用 netcore 2.2 作为目标框架,因此需要安装 netcore 2.2 SDK 才能运行。
- 本文作者: Newbe36524
- 本文链接: http://www.newbe.pro/2019/09/10/Others/Try-More-On-IEnumerable-2/
- 版权声明: 本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
试试 IEnumerable 的另外 6 个小例子的更多相关文章
- 试试 IEnumerable 的 10 个小例子
IEnumerable 接口是 C# 开发过程中非常重要的接口,对于其特性和用法的了解是十分必要的.本文将通过10个小例子,来熟悉一下其简单的用法. 全是源码 以下便是这10个小例子,响应的说明均标记 ...
- Runtime的几个小例子(含Demo)
一.什么是runtime(也就是所谓的“运行时”,因为是在运行时实现的.) 1.runtime是一套底层的c语言API(包括很多强大实用的c语言类型,c语言函数); [runti ...
- 2、Lucene 最简单的使用(小例子)
在了解了Lucene以后,我打算亲手来做一个Lucene的小例子,这个例子只是Lucene最简单的应用:使用Lucene实现标准的英文搜索: 1.下载Lucene 下载Lucene,到Lucene的官 ...
- vuex2.0+两个小例子
首先vuex概念比较多,一定要搞懂里面的概念,可以参考官网Vuex2.0概念,我写此文的目的是希望能对前端爱好者提供个参考,加深对vuex2.0各核心概念的理解. 废话少说,直接上干货.这是官网上的一 ...
- 一个spring boot集成dubbo的小例子
请移步github,介绍和代码均在上面了:https://github.com/wuxun1997/voicebox 这里再多说两句.github上的这个小例子默认使用组播作为注册中心,你也可以把组播 ...
- Vuex2.0边学边记+两个小例子
最近在研究Vuex2.0,搞了好几天终于有点头绪了. 首先vuex概念比较多,一定要搞懂里面的概念,可以参考官网Vuex2.0概念,我写此文的目的是希望能对前端爱好者提供个参考,加深对vuex2.0各 ...
- C#中把任意类型的泛型集合转换成SQLXML数据格式的小例子
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.D ...
- python2.7练习小例子(十)
10):古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少? 程序分析:兔子的规律为数列1,1 ...
- springmvc入门的第一个小例子
今天我们探讨一下springmvc,由于是初学,所以简单的了解一下 springmvc的流程,后续会持续更新... 由一个小例子来简单的了解一下 springmvc springmvc是spring框 ...
随机推荐
- 创建String对象过程的内存分配
转载自 https://blog.csdn.net/xiabing082/article/details/49759071 常量池(Constant Pool):指的是在编译期被确定 ...
- 全球十大OTA 谁能有一席之地?
全球十大OTA 谁能有一席之地? http://www.traveldaily.cn/article/78381/1 2014-03-05 来源:i黑马 随着旅游行业日新月异的发展,在线旅游网站的出现 ...
- java字符串详解
一.String 类的定义 public final class String implements java.io.Serializable, Comparable<String>, C ...
- hadoop2.7+spark2.2+zookeeper3.4.简单安装
1.zookeeper的安装##配置/etc/hosts192.168.88.130 lgh192.168.88.131 lgh1192.168.88.132 lgh2 ##安装java8 解压配置环 ...
- 从IDEA角度来看懂UML图
前言 我们目前已经学习了设计模式的7种设计原则.下面本该是直接进入具体的设计模式系列文章. 但是呢在我们学习设计模式之前我们还是有必要了解一下uml图.因为后续的设计模式文章不出意外应该会很多地方使用 ...
- SpringBoot学习------SpringBoot使用Thymleaf模块访问不了静态页面
SpringBoot使用Thymleaf模块访问不了静态页面 最近学习SpringBoot的过程中使用了Thymeleaf模块引擎,页面发送请求后老是无法显示静态页面,所有的步骤都是参考资料来执行,自 ...
- Springboot源码分析之jar探秘
摘要: 利用IDEA等工具打包会出现springboot-0.0.1-SNAPSHOT.jar,springboot-0.0.1-SNAPSHOT.jar.original,前面说过它们之间的关系了, ...
- 如何使用WorkManager执行后台任务(下)
0x00 WorkManager的高级用法 在上一文中已经了解到 WorkManager的基本用法之后,今天来看看它的一些高级用法: 链式任务调用 唯一任务序列 传递参数和获取返回值 0x01 链式任 ...
- hmac模块和hashlib模块
hmac模块和hashlib模块 一.hash是什么 hash是一种算法(Python3.版本里使用hashlib模块代替了md5模块和sha模块,主要提供 SHA1.SHA224.SHA256. ...
- Java 8 为什么会引入lambda 表达式?
Java 8 为什么会引入lambda ? 在Java8出现之前,如果你想传递一段代码到另一个方法里是很不方便的.你几乎不可能将代码块到处传递,因为Java是一个面向对象的语言,因此你要构建一个属于某 ...