有效地处理数据时当今程序设计语言和框架的一个任务。.NET拥有一个精心构建的集合类系统,它利用迭代器的功能实现对数据的顺序访问。

惰性枚举是一个迭代方法,其核心思想是只在需要的时候才去读取数据。这个思想保证了任何迭代算法都十分有效,同时又可以灵活地根据需要读取任意多的数据,而且不会造成过多的开销。

C#函数式程序设计之枚举元素

.NET集合类型的基础是一个名为IEnumberable的接口,以下就是这个接口的声明:

public interface IEnumerable
{
IEnumerator GetEnumerator();
}

实际上IEnumberable接口只允许程序员做一件事:查询类的另一个接口IEnumerator:

public interface IEnumerator
{
object Current { get; }
bool MoveNext();
void Reset();
}

下面这个类事实上是根据IEnumberable和IEnumerator接口声明的迭代模式的最简实现形式:

public class EndlessListWithoutInterfaces
{
public EndlessListWithoutInterfaces GetEnumerator()
{
return this;
} public bool NoveNext()
{
return true;
} public object Current
{
get
{ return "Something"; }
}
}

在EndlessListWithoutInterfaces类中也可以使用C#的foreach结构:

var list = new EndlessListWithoutInterfaces();
foreach (var item in list)
{
Console.WriteLine(item);
}

此实现极其基本,因此这将导致一个无限循环。

下面是EndlessList类的实现,这个类使用IEnumerator和IEnumberable两个类,这两个类相互嵌套,尽管它们不需要里外嵌套。但实际上,在IEnumberable类中嵌套IEnumerator类是一个常见模式:

    public class EndlessList:IEnumerable
{
public class Enumerator:IEnumerator
{
int val = -1; public object Current
{
get
{ return val; }
} public bool NoveNext()
{
val++;
return true;
} public void Reset()
{
val = -1;
}
} public IEnumerator GetEnumerator()
{
return new Enumerator();
}
}

值序列的这种实现模式是非常灵活的,而且功能也强大。

C#函数式程序设计之迭代器函数的实现

C#2.0版本中引入了迭代器。它允许程序员创建IEnumerator/IEnumberable组合的实现,而无须手动实现其中的任意一个接口。除了支持非泛型接口外,它还支持泛型接口,因此可以只实现IEnumerator。

通常为了应用这个功能,只需要定义一个返回某个类型值的函数。编译器为了使用它的转换操作而采用的第二个准则是在函数中至少使用几个特殊关键字中的一个。最常用的是yield return语句。例如,前面的EndlessList例子可以由如下的一个C#迭代器实现:

        public static IEnumerable<int> EndlessListFunc()
{
int val = 0;
while (true)
yield return val++;
}

此函数的返回类型是一个IEnumberable<int>,因此在原本使用一个能实现此接口的类实例的地方现在可以使用这个函数。下面几个语句可以迭代访问EndlessListFunc序列:

var list = EndlessListFunc();
foreach (var item in list)
{
Console.WriteLine(item);
}

IEnumerator/IEnumberable两个接口使用的整个声明模式与惰性有关,即只是在需要时才读取数据。这种模式的优点是:一个序列只是整个算法的一个很少但非常灵活的部分,序列也无需对它的将来使用进行任何假设。

下面这段代码是一个迭代器的实际例子,利用Twitter流的一个搜索,从Web服务读取数据。从Web服务读取数据也采用惰性方法,因为它使用了自动分页技术。迭代器根据需要读取新页,把长度未知的字符串序列返回给调用者。调用者按照自己喜欢的方式处理这个序列:

        private static void TwitterSearchIterator()
{
foreach (var tweet in GetTweets("#msdn").Take(10))
Console.WriteLine(tweet);
} private static IEnumerable<string> GetTweets(string searchString)
{
var url = "http://search.twitter.com/search.atom?q=";
int page = 1;
var escapedSearchString = searchString.Replace("@", "%40").Replace("#", "%23");
XNamespace ns = "http://www.w3.org/2005/Atom"; while (true)
{
var doc = XDocument.Load(String.Format("{0}{1} & page={2}", url, escapedSearchString, page));
var entries = doc.Root.Elements(ns + "entry");
if (entries.Count() == 0)
yield break;
foreach (var entry in entries)
yield return
entry.Element(ns+"author").Element(ns+"name").Value+": "+WebUtility.HtmlDecode(entry.Element(ns+"title").Value;
page++;
}
}

迭代器不仅可以返回前面例子使用的IEnumberable和IEnumberable<T>接口,还可以返回IEnumerator和IEnumerator<T>接口。

C#函数式程序设计之链式迭代器

很容易把函数形式的迭代器链接在一起,并通过它们创建复杂的处理流程。这个思想广泛用于LINQ,这是函数式程序设计一个最重要的概念。

前面介绍了EndlessListFunc迭代器,它可以返回一个无穷的整数值序列,处理序列的迭代器函数可以与这个迭代器一起使用。例如下例是Take函数的实现,它通过参数接受一个证书序列和一个计数值,且只返回源序列中的前几个元素:

public static IEnumerable<int> Take(int count, IEnumerable<int> source)
{
int used = 0;
foreach (var item in source)
if (count > used++)
yield return item;
else
yield break;
}

可以把Take与EndlessListFunc一起使用:

var FiveElements = Take(5, EndlessListFunc());
foreach (var item in FiveElements)
Console.WriteLine(item);

这个例子很好地说明了如何把迭代器当成模块使用。

用在链式迭代器中的第二种类型函数是使用实际内容的函数,比如执行数值计算。

C#函数式程序设计之惰性列表工具——迭代器的更多相关文章

  1. C#函数式程序设计之函数、委托和Lambda表达式

    C#函数式程序设计之函数.委托和Lambda表达式 C#函数式程序设计之函数.委托和Lambda表达式   相信很多人都听说过函数式编程,提到函数式程序设计,脑海里涌现出来更多的是Lisp.Haske ...

  2. C#函数式程序设计之局部套用与部分应用

    函数式设计的核心与函数的应用以及函数如何作为算法的基本模块有关.利用局部套用技术可以把所有函数看成是函数类的成员,这些函数只有一个形参,有了局部套用,才有部分应用.部分应用是使函数模块化成为可能的两个 ...

  3. C#函数式程序设计之用闭包封装数据

    如果一个程序设计语言能够用高阶函数解决问题,则意味着数据作用域问题已十分突出.当函数可以当成参数和返回值在函数之间进行传递时,编译器利用闭包扩展变量的作用域,以保证随时能得到所需要的数据. C#函数式 ...

  4. C#函数式程序设计之泛型

    Intellij修改archetype Plugin配置 2014-03-16 09:26 by 破狼, 204 阅读, 0 评论,收藏, 编辑 Maven archetype plugin为我们提供 ...

  5. C#函数式程序设计之泛型(上)

    在面向对象语言中,我们可以编写一个元素为某个专用类型(可能需要为此创建一个ListElement)的List类,或者使用一个非常通用.允许添加任何类型元素的基类(在.NET中,首先想到的是System ...

  6. C#函数式程序设计之代码即数据

    自3.5版本以来,.NET以及微软的.NET语言开始支持表达式树.它们为这些语言的某个特定子集提供了eval形式的求值功能.考虑下面这个简单的Lambda表达式: Func<int, int, ...

  7. C#函数式程序设计之泛型(下)

    C#函数式程序设计之泛型(下)   每当使用泛型类型时,可以通过where字句对泛型添加约束: + 这个例子直观地声明了一个约束:类型T必须与ListItem<string>相匹配.泛型类 ...

  8. QMUI UI库 控件 弹窗 列表 工具类 MD

    Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...

  9. 基于 Generator 和 Iterator 的惰性列表

    初识 Lazy List 如果有了解过 Haskell 的朋友,对下面的这些表达一定不陌生 repeat 1 -- => [1, 1, 1, 1, 1,...] cycle "abc& ...

随机推荐

  1. 【UVA】1589 Xiangqi(挖坑待填)

    题目 题目     分析 无力了,noip考完心力憔悴,想随便切道题却码了250line,而且还是错的,知道自己哪里错了,但特殊情况判起来太烦了,唯一选择是重构,我却没有这勇气. 有空再写吧,最近真的 ...

  2. 显示本月日历demo

    import java.text.DateFormatSymbols; import java.util.Calendar; import java.util.GregorianCalendar; p ...

  3. OD 实验(十六) - 从对话框入手对程序的逆向

    对话框: 对话框从类型上分为两类:modal 对话框和 modeless 对话框,就是模态对话框和非模态对话框,也有叫成模式和非模式 模态对话框不允许用户在不同窗口间进行切换,非模态对话框允许用户在不 ...

  4. cinder backup ceph的配置和使用

    Backup 是将 volume 备份到别的地方(备份设备),将来可以通过 restore 操作恢复. 初看 backup 功能好像与 snapshot 很相似,都可以保存 volume 的当前状态, ...

  5. Shift Operations on C

    The C standard doesn't precisely define which type of right shift should be used. For unsigned data, ...

  6. <转>Python3.x和Python2.x的区别

    这个星期开始学习Python了,因为看的书都是基于Python2.x,而且我安装的是Python3.1,所以书上写的地方好多都不适用于Python3.1,特意在Google上search了一下3.x和 ...

  7. iframe callback方式文件上传

    1.前端default.html <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "h ...

  8. django之管理静态文件

    管理静态文件 项目中的CSS.图片.js都是静态文件 配置静态文件 在settings 文件中定义静态内容 STATIC_URL = '/static/' STATICFILES_DIRS = [ o ...

  9. leetcode529

    public class Solution { //DFS public char[,] UpdateBoard(char[,] board, int[] click) { ), n = board. ...

  10. Hexo+Github/Coding免费搭建个人博客网站

    体验更优排版请移步原文:http://blog.kwin.wang/other/hexo-github-build-blog.html 很早之前就想搭建一个属于自己的博客网站,一方面是给自己做笔记,把 ...