有效地处理数据时当今程序设计语言和框架的一个任务。.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. WordSmith2013-7-31

    WordSmith Good Evening Ladies and Gentlemen,I’am Jason,I’m pleasured  to be wordsmith tonight. First ...

  2. touch python

    一  使用while循环输出 1 2 3 4 5 6 8 9 10. i=0 while i<10: i=i+1 if i == 7: continue print(i) 二  求 1-100所 ...

  3. nginx代理socket tcp/udp

    准备一台linux服务器.nginx官网:http://nginx.org/ .在网上搜到大致用的是 ngx_stream_core_module 这个模块,这里你也可以关注一下官方文档中的其他模块都 ...

  4. C++虚函数表理解

    一,思维模式图 二,代码验证 class A { public: A(int x) { fProtected = x; } float GetFProtected() { return fProtec ...

  5. .net core 2.0的一次奇特经历

    环境:.net core SDK版本 2.0.0-preview1-005977 VS 2017 version 15.3.0 preview 3.0 问题描述:今天在迁移Job的项目中,中午吃饭的时 ...

  6. ES6中新添加的Array.prototype.fill

    用法 array.fill(start=0, end=this.length) 示例 [1, 2, 3].fill(4) // [4, 4, 4] [1, 2, 3].fill(4, 1) // [1 ...

  7. [iOS]隐藏导航栏把右滑退出操作保留

    项目因为用到上面导航栏样式多变,就隐藏了导航栏自己用View代替了,但手势却不见了,后来发现问题解决.操作如下: 千万不要取消 Shows Navigation Bar 这个选项否则手势会消失 应该是 ...

  8. mysql varchar类型转换int类型

    select * from gyzd_yysinfo order by cast(yysid as SIGNED INTEGER) 或者 select * from gyzd_yysinfo orde ...

  9. Kafka存储机制(转)

    转自:https://www.cnblogs.com/jun1019/p/6256514.html Kafka存储机制 同一个topic下有多个不同的partition,每个partition为一个目 ...

  10. ajax获取json数据为undefined--原因解析

    解决办法:var dataObj=eval("("+data+")");//转换为json对象 问题: 1. 碰到一个问题ajax成功获取json数据后,取值显 ...