探秘C#中的yield关键字
在"C#中,什么时候用yield return"中,我们了解到:使用yield return返回集合,不是一次性加载到内存中,而是客户端每调用一次就返回一个集合元素,是一种"按需供给"。本篇来重温yield return的用法,探秘yield背后的故事并自定义一个能达到yield return相同效果的类,最后体验yield break的用法。
□ 回顾yield return的用法
以下代码创建一个集合并遍历集合。
class Program{static Random r = new Random();static IEnumerable<int> GetList(int count){List<int> list = new List<int>();for (int i = 0; i < count; i++){list.Add(r.Next(10));}return list;}static void Main(string[] args){foreach(int item in GetList(5))Console.WriteLine(item);Console.ReadKey();}}
使用yield return也能获得同样的结果。修改GetList方法为:
static IEnumerable<int> GetList(int count){for (int i = 0; i < count; i++){yield return r.Next(10);}}
通过断点调试发现:客户端每显示一个集合中的元素,都会到GetList方法去获取集合元素。
□ 探密yield
使用yield return获取集合,并遍历。
class Program{public static Random r = new Random();static IEnumerable<int> GetList(int count){for (int i = 0; i < count; i++){yield return r.Next(10);}}static void Main(string[] args){foreach(int item in GetList(5))Console.WriteLine(item);Console.ReadKey();}}
生成项目,并用Reflector反编译可执行文件。在.NET 1.0版本下查看GetList方法,发现该方法返回的是一个GetList类的实例。原来yield return是"语法糖",其本质是生成了一个GetList的实例。
那GetList实例是什么呢?点击Reflector中<GetList>链接查看。
○ 原来GetList类实现了IEnumerable和IEnumerator的泛型、非泛型接口
○ yield return返回的集合之所以能被迭代、遍历,是因为GetList内部有迭代器
○ yield return之所以能实现"按需供给",是因为GetList内部有一个_state字段记录这上次的状态
接下来,就模拟GetList,我们自定义一个GetRandomNumbersClass类,使之能达到yield return相同的效果。
using System;using System.Collections;using System.Collections.Generic;namespace ConsoleApplication2{class Program{public static Random r = new Random();static IEnumerable<int> GetList(int count){GetRandomNumbersClass ret = new GetRandomNumbersClass();ret.count = count;return ret;}static void Main(string[] args){foreach(int item in GetList(5))Console.WriteLine(item);Console.ReadKey();}}class GetRandomNumbersClass : IEnumerable<int>, IEnumerator<int>{public int count;//集合元素的数量public int i; //当前指针private int current;//存储当前值private int state;//保存遍历的状态#region 实现IEnumerator接口public int Current{get { return current; }}public bool MoveNext(){switch (state){case 0: //即为初始默认值i = 0;//把指针调向0goto case 1;break;case 1:state = 1;//先设置原状态if (!(i < count))//如果指针大于等于当前集合元素数量{return false;}current = Program.r.Next(10);state = 2; //再设置当前状态return true;break;case 2: //再次遍历如果state值为2i++;//指针再移动一位goto case 1;break;}return false;}//被显式调用的属性object IEnumerator.Current{get { return Current; }}public void Reset(){throw new NotImplementedException();}public void Dispose(){}#endregion#region 实现IEnumerable的泛型和非泛型public IEnumerator<int> GetEnumerator(){return this;}//被显式调用的属性IEnumerator IEnumerable.GetEnumerator(){return GetEnumerator();}#endregion}}
关于GetRandomNumbersClass类:
○ count表示集合的长度,可以在客户端赋值。当调用迭代器的MoveNext方法,需要把count和当前位置比较,以决定是否可以再向前移动。
○ 字段i相当于索引,指针每次移动一位,i需要自增1
○ current表示当前存储的值,外部通过IEnumerator.Current属性访问
迭代器的MoveNext方法是关键:
○ state字段是整型,表示产生集合过程中的3种状态
○ 当state为0的时候,说明是初始状态,把索引位置调到0,并跳转到state为1的部分
○ 当state为1的时候,首先把状态设置为1,然后判断索引的位置有没有大于或等于集合的长度,接着产生集合元素,把state设置为2,并最终返回true
○ 当sate为2的时候,也就是迭代器向前移动一位,再次执行MonveNext方法的时候,跳转到state为2的语句块部分,把索引位置自增1,再跳转到state为1的语句块中,产生新的集合元素
○ 如此循环
□ yield break的用法
假设在一个无限循环的环境中获取一个int类型的集合,在客户端通过某个条件来终止循环。
class Program{static Random rand = new Random();static IEnumerable<int> GetList(){while (true){yield return rand.Next(100);}}static void Main(string[] args){foreach (int item in GetList()){if (item%10 == 0){break;}Console.WriteLine(item);}Console.ReadKey();}}
以上,当集合元素可以被10整除的时候,就终止循环。终止循环的时机是在循环遍历的时候。
如果用yield break,就可以在获取集合的时候,当符合某种条件就终止获取集合。
class Program{static Random rand = new Random();static IEnumerable<int> GetList(){while (true){int temp = rand.Next(100);if (temp%10 == 0){yield break;}yield return temp;}}static void Main(string[] args){foreach (int item in GetList()){Console.WriteLine(item);}Console.ReadKey();}}
总结:
○ yield return能返回一个"按需供给"的集合
○ yield return是"语法糖",其背后是一个实现了IEnuerable,IEnumerator泛型、非泛型接口的类,该类维护着一个状态字段,以保证yield return产生的集合能"按需供给"
○ yield break配合yield return使用,当产生集合达到某种条件的时候使用yield break,以终止继续创建集合
探秘C#中的yield关键字的更多相关文章
- .NET中的yield关键字
浅谈yield http://www.cnblogs.com/qlb5626267/archive/2009/05/08/1452517.html .NET中yield关键字的用法 http://bl ...
- C#2.0中使用yield关键字简化枚举器的实现
我们知道要使用foreach语句从客户端代码中调用迭代器,必需实现IEnumerable接口来公开枚举器,IEnumerable是用来公开枚举器的,它并不实现枚举器,要实现枚举器必需实现IEnumer ...
- 解析Python中的yield关键字
前言 python中有一个非常有用的语法叫做生成器,所利用到的关键字就是yield.有效利用生成器这个工具可以有效地节约系统资源,避免不必要的内存占用. 一段代码 def fun(): for i i ...
- 深入理解python中的yield关键字
想必大家都看过这样的代码: 上面的这段代码会计算0-9的平方并打印出来. 那么问题来了,这段代码和我们要说的东西有什么区别呢? 这里的关键字,yield,我在前面的文章里已经发过了.那么yield是什 ...
- python中的yield关键字
yield关键字一直困扰了我很久,一直也没有弄明白,现在将暂时理解的yield记录如下,供参考: 关键词:可迭代对象,生成器,迭代器 一.可迭代对象: 可迭代对象:可迭代对象是一个泛称,只要可以用fo ...
- Java中的yield关键字的简单讲解
Thread.yield()方法作用是:暂停当前正在执行的线程对象,并执行其他线程. yield()应该做的是让当前运行线程回到可运行状态,以允许具有相同优先级的其他线程获得运行机会.因此,使用yie ...
- C#中的yield关键字
迭代器,是一个连续的集合,出现多个yield return其实就是将这多个的yield return元素按照出现的顺序存储在迭代器的集合中而已.形如下面的形式: public class CityCo ...
- yield 关键字和迭代器
一般使用方法 yield 关键字向编译器指示它所在的方法是迭代器块 在迭代器块中,yield 关键字与 return 关键字结合使用,向枚举器对象提供值. 这是一个返回值,例如,在 forea ...
- C# 基础小知识之yield 关键字 语法糖
原文地址:http://www.cnblogs.com/santian/p/4389675.html 对于yield关键字我们首先看一下msdn的解释: 如果你在语句中使用 yield 关键字,则意味 ...
随机推荐
- ****jQuery - 设置HTML内容和属性
设置内容 - text().html() 以及 val() 我们将使用前一章中的三个相同的方法来设置内容: text() - 设置或返回所选元素的文本内容 html() - 设置或返回所选元素的内容( ...
- bms_output.put_line使用方法
https://blog.csdn.net/sxww321/article/details/4020300
- 两种方法设置nginx并发限制下面的白名单策略
前言: 今天,公司主站突然出现IDE创建应用没反应的问题,经过预发布环境.非代理环境下面的服务测试,均没有问题,定位问题出现在前端.而我们前端有两层代理,一是青松抗D系统,一是我们自己的nginx代理 ...
- ASP.NET Web API 2:创建API帮助页面
当你新建了一个web API服务之后,再建一个API帮助页面是很有好处的,这样其他开发人员就会很清楚地知道如何调用你的API接口.你可以选择自己手工建立,但是如果能自动生成岂不是更好.为了简 ...
- Ionic入门三:列表
列表是一个应用广泛的界面元素,在所有移动app中几乎都会使用到. 列表可以是基本文字.按钮,开关,图标和缩略图等. 列表项可以是任何的HTML元素.容器元素需要list类,每个列表项需要使用item类 ...
- Swift2.0语言教程之Swift2.0语言中的标准函数
Swift2.0语言教程之Swift2.0语言中的标准函数 Swift2.0中的标准函数 函数除了可以根据参数列表的有无分为无参函数和有参函数,还可以从定义角度分为用户自定义函数和标准函数两种.以上的 ...
- web服务端安全之暴力破解
一.暴力破解 指攻击者通过遍历或字典的方式,向目标发起大量请求,通过判断返回数据包的特征来找出正确的验证信息,从而绕过验证机制. 二.常见场景 用户登录处的账号密码暴力破解: 人机验证机制容易绕过,如 ...
- python opencv3 直线检测
git:https://github.com/linyi0604/Computer-Vision # coding:utf8 import cv2 import numpy as np # 读入图像 ...
- php 简单计算权重的方法(适合抽奖类的应用)
//简单权重计算器 $data222=array( 0=>array('id'=>1,'name'=>'一等奖','weight'=>'3'), 1=>a ...
- SCC缩点
int V; //顶点数量 vector<int> G[max_v]; //图的邻接表表示方法 vector<int> rG[max_v]; //把边反向建的图 vector& ...