C#函数式程序设计之惰性列表工具——迭代器
有效地处理数据时当今程序设计语言和框架的一个任务。.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#函数式程序设计之惰性列表工具——迭代器的更多相关文章
- C#函数式程序设计之函数、委托和Lambda表达式
C#函数式程序设计之函数.委托和Lambda表达式 C#函数式程序设计之函数.委托和Lambda表达式 相信很多人都听说过函数式编程,提到函数式程序设计,脑海里涌现出来更多的是Lisp.Haske ...
- C#函数式程序设计之局部套用与部分应用
函数式设计的核心与函数的应用以及函数如何作为算法的基本模块有关.利用局部套用技术可以把所有函数看成是函数类的成员,这些函数只有一个形参,有了局部套用,才有部分应用.部分应用是使函数模块化成为可能的两个 ...
- C#函数式程序设计之用闭包封装数据
如果一个程序设计语言能够用高阶函数解决问题,则意味着数据作用域问题已十分突出.当函数可以当成参数和返回值在函数之间进行传递时,编译器利用闭包扩展变量的作用域,以保证随时能得到所需要的数据. C#函数式 ...
- C#函数式程序设计之泛型
Intellij修改archetype Plugin配置 2014-03-16 09:26 by 破狼, 204 阅读, 0 评论,收藏, 编辑 Maven archetype plugin为我们提供 ...
- C#函数式程序设计之泛型(上)
在面向对象语言中,我们可以编写一个元素为某个专用类型(可能需要为此创建一个ListElement)的List类,或者使用一个非常通用.允许添加任何类型元素的基类(在.NET中,首先想到的是System ...
- C#函数式程序设计之代码即数据
自3.5版本以来,.NET以及微软的.NET语言开始支持表达式树.它们为这些语言的某个特定子集提供了eval形式的求值功能.考虑下面这个简单的Lambda表达式: Func<int, int, ...
- C#函数式程序设计之泛型(下)
C#函数式程序设计之泛型(下) 每当使用泛型类型时,可以通过where字句对泛型添加约束: + 这个例子直观地声明了一个约束:类型T必须与ListItem<string>相匹配.泛型类 ...
- QMUI UI库 控件 弹窗 列表 工具类 MD
Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...
- 基于 Generator 和 Iterator 的惰性列表
初识 Lazy List 如果有了解过 Haskell 的朋友,对下面的这些表达一定不陌生 repeat 1 -- => [1, 1, 1, 1, 1,...] cycle "abc& ...
随机推荐
- Apache中按天分割日志(Windows)
网上很多资料都有对Apache的access.log按天生成的方法,但在Windows server下稍有不同: 1.打开httpd.conf配置文件找到: CustomLog "logs/ ...
- 升级phpcms的ckeditor编辑器
首先说明一下为什么升级?网上很多人升级成了ueditor,可从fckedotror 到 ckeditor,我个人都是比较喜欢的,特别是开放式的插件方式.另外一个就是至少要懂得升级和插件的开发,这样也能 ...
- zabbix agent主动模式与proxy模式,实现公网zabbix监控私网客户机
zabbix agent主动模式,实现公网zabbix监控私网客户机 zabbix_server端当主机数量过多的时候,由Server端去收集数据,Zabbix会出现严重的性能问题,主要表现如下: ...
- 优化笔记:pfyhparopenfundinfotest_D_20140916.gz
性能瓶颈在函数的乱用.原代码黄色部分. 12分钟->35秒 ------------------------------------------------------------------- ...
- 接口测试基础operation
Jmeter接口测试 一般分五个步骤: (1)添加线程组 (2)添加http请求 (3)在http请求中写入接入url.路径.请求方式和参数 (4)添加查看结果树 (5)调用接口.查看返回值
- SlidingMenu Demo
参考:http://www.krislq.com/2013/03/android_case_slidingmenu_fragment/ 我下载了它的例子,然后自己重写了一下,运行时总报错,原来是sup ...
- 关于服务器raid的一个记录
今天下午,在装操作系统的时候,特意的测试了下raid1的性能. 1. 开启操作系统 直接正常开启操作系统,操作系统的硬盘做的是raid1,从而数据写俩份,从而在损坏一张盘之后,另外一张盘并不会收到影响 ...
- Python格式符说明
格式化输出 例如我想输出 我的名字是xxxx 年龄是xxxx name = "Lucy"age = 17print("我的名字是%s,年龄是%d"%(name, ...
- TreeView绑定
) { return; } foreach (NodeInfo info in list) ...
- Unity iOS 项目的一种性能评测方法
[Unity iOS 项目的一种性能评测方法]