彻底搞懂C#之Yield Return语法的作用和好处
还是和以前一样,我先上代码,请大家先拿到我的代码或者你跟着敲,运行看效果,以及理解每行带代码的作用。 我们要带着为什么要用Yield这个关键字,不用可以吗这个目的去学知识,我相信会更加的有意思。 首先我贴出平时正常输出偶数集合的办法 --------------------- 本文来自 牛掰是怎么形成的 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/qq_33060405/article/details/78484825?utm_source=copy
/* * * 学习Yield Return 语法 * 使用两个方法,显示1 - 100之间的全部偶数 * * */ using System; using System.Collections.Generic; namespace YieldReturn语法解析 { class Program { static private List<int> _numArray; //用来保存1-100 这100个整数 Program() //构造函数。我们可以通过这个构造函数往待测试集合中存入1-100这100个测试数据 { _numArray = new List<int>(); //给集合变量开始在堆内存上开内存,并且把内存首地址交给这个_numArray变量 for (int i = 1; i <= 100; i++) { _numArray.Add(i); //把1到100保存在集合当中方便操作 } } static void Main(string[] args) { new Program(); TestMethod(); } //测试求1到100之间的全部偶数 static public void TestMethod() { foreach (var item in GetAllEvenNumber()) { Console.WriteLine(item); //输出偶数测试 } } //测试我们正常情况下拿到全部偶数的方法 static IEnumerable<int> GetAllEvenNumber() { List<int> result = new List<int>(); //开集合内存存偶数用 foreach (int num in _numArray) { if(num % 2 == 0) //判断是不是偶数 { //yield return num; result.Add(num); //存入集合 } } //返回偶数集合变量 可能有人会觉得奇怪返回类型不是List<int>这样可以吗 //这个就要回到我们的里氏替换原则了,子类是可以替换父类的,也就是当父类用 //比如我这个方法是想得到IEnumerable<int> 类型变量,但是我给了List<int>类型变量 //注意List<int> 是继承 IEnumerable<int> 的,什么意思当我们把子类当父类使用, //那么大才小用,因为子类很多都是继承父亲,你自身增加很多字段或者方法,这样就不能用了。 return result; //yield break; } } }
贴出使用Yiele Return办法输出耦合集合
/* * * 学习Yield Return 语法 * 使用两个方法,显示1 - 100之间的全部偶数 * * */ using System; using System.Collections.Generic; namespace YieldReturn语法解析 { class Program { static private List<int> _numArray; //用来保存1-100 这100个整数 Program() //构造函数。我们可以通过这个构造函数往待测试集合中存入1-100这100个测试数据 { _numArray = new List<int>(); //给集合变量开始在堆内存上开内存,并且把内存首地址交给这个_numArray变量 for (int i = 1; i <= 100; i++) { _numArray.Add(i); //把1到100保存在集合当中方便操作 } } static void Main(string[] args) { new Program(); TestMethod(); } //测试求1到100之间的全部偶数 static public void TestMethod() { foreach (var item in GetAllEvenNumber()) { Console.WriteLine(item); //输出偶数测试 } } //测试我们使用Yield Return情况下拿到全部偶数的方法 static IEnumerable<int> GetAllEvenNumber() { foreach (int num in _numArray) { if(num % 2 == 0) //判断是不是偶数 { yield return num; //返回当前偶数 } } yield break; //当前集合已经遍历完毕,我们就跳出当前函数,其实你不加也可以 //这个作用就是提前结束当前函数,就是说这个函数运行完毕了。 } } }
大家测试了2个代码结果没,是不是都可以正确拿到全部偶数集合,具体我需要你们测,这样进步快,才会真是学会。只看不练假把戏。 现在我们说他们的区别: 这个才是真正要学的地方 我们需要下断点
我希望你是在断点调试,具体就贴太多,你会发现代码跑到Yield return num的时候,当前函数就会结束,并把这个num交给这里,我先上图
最终会给item变量。 然后输出,然后在执行 GetAllEvenNumber()方法,和上次得到偶数是一样。 我们发现这个Yield Return是可以让当前函数的进程状态切换到阻塞状态,然后去选择了把cpu交给当前的出进程,这样就转而执行调用方函数。 (补充个小知识点其实我们写的程序加入到内存中,并不定就是一个进程,我们会根据情况分成几个子进程去干活,方便操作系统去管理以及多道程序运行在内存,提高计算机资源的利用率) 这样有个好处,我们假如有1000000个数据,我们需要得到里面的耦合,当我们通过这个方法得到一个耦合会立马显示在控制台上。而不是等很久也就把全部偶数都查找到存入集合当中,然后在一一遍历输出。 这个好处是很大的。比如我们用户可能就看数据开始肯定不是全部值需要部分就可以,看完这些在看后面的,这样数据会让觉得显示没有延迟。 下面我们来断点调一般的查找偶数的函数
我们发现这个循环没有跑完是不会退出当前函数的,也就是必须要查找全部的偶数才可以。这样假如我们很多数据。就会一直等这个数据才可以拿到这个数据去显示给用户看。 这样就麻烦了。 因此我们下个结论:Yield Return关键字的作用就是退出当前函数,并且会保存当前函数执行到什么地方,也就上下文。你发现没下次执行这个函数上次跑来的代码是不会重复执行的, 但是你一般的return result 假如你在循环体提前return ,下面调这个函数是会从第一步开始重新执行的。不会记录上次执行的地方。