`
RednaxelaFX
  • 浏览: 3049624 次
  • 性别: Icon_minigender_1
  • 来自: 海外
社区版块
存档分类
最新评论

C#中的generator不会被立即执行……

    博客分类:
  • C#
阅读更多
读了Eric Lippert的这帖:High maintenance之后,心里一阵寒——我自己写的代码里就有几乎一模一样的逻辑。

Eric的帖里举的不好的例子:
public static class StreamReaderExtensions
{
    public static IEnumerable<string> Lines(this StreamReader reader)
    {
        if (reader== null)
            throw new ArgumentNullException("reader");
        reader.BaseStream.Seek(0, SeekOrigin.Begin);
        string line;
        while ((line = reader.ReadLine()) != null)
            yield return line;
    }
}


我的代码:
/// <summary>
/// Reads the stream as lines of strings from the current
/// position to the end of the stream.
/// </summary>
/// <param name="reader">the stream to read from</param>
/// <returns>An enumerator of the lines of strings.</returns>
public static IEnumerable<string> Lines( this StreamReader reader ) {
    for ( var line = reader.ReadLine( ); null != line; line = reader.ReadLine( ) ) {
        yield return line;
    }
}

事实上我的还更糟糕一些,没做null检查。写扩展方法的时候我总是写着写着就忘了:扩展方法不是成员方法,作为“this”的参数需要做null检查。

Eric对他所举的例子指出了几个不足:
1、null检查不是在方法调用时执行,而是在迭代器第一次移动的时候才执行的
2、while循环在一行塞了太多逻辑,可以分开来写;
3、通过BaseStream来改变了底下的流的位置,不符合习惯上对流的使用方式;
4、调用者需要知道太多细节,例如必须知道何时这个迭代器结束了,在结束后由调用者来关闭流;
……等。

第4点我在写我那个程序时也知道这里有问题,但没想出好的解决办法。假如我就是要特意把流设置到某个位置之后再调用Lines(),该怎么办呢?或者说假如原本就不存在一个实际文件(例如stdin),那就无法用文件名来指定参数了,又怎么办呢?
于是在Stream的ownership上我还是得再思考一下才行。这篇主要是要提醒自己C#的generator的一个重要特性:

带有yield关键字的方法(也就是generator),其方法体在generator被调用时是不会被执行的;只有当其返回的iterator的MoveNext()方法被初次调用时才会执行。

举例来说,这样:
using System;
using System.Collections.Generic;

static class Program {
    static IEnumerable<int> FooGenerator( int max ) {
        Console.WriteLine( "Starting enumerator" );
        for ( var i = 0; i < max; ++i ) {
            yield return i;
        }
    }

    static void Main( string[ ] args ) {
        var list = FooGenerator( 3 );
        Console.WriteLine( "After Foo(), before foreach" );
        foreach ( var i in list ) {
            Console.WriteLine( i );
        }
    }
}

输出的结果会是:
After Foo(), before foreach
Starting enumerator
0
1
2


了解C#的generator的实现方式就不难理解这个行为的来源。C#的generator实际上是个由编译器自动生成的实现了IEnumerable<T>接口的有限状态机。也就是说generator里实际上只有一个return new ...,或许还会有些参数赋值,却没有任何别的内容。源码里写在generator里的逻辑都生成到了那个有限状态机对象里,也就是外界看到的迭代器里。如果要对参数做检查,恐怕还就是Eric说的,提供一个公有方法作为接口,在里面检查参数的正确性,然后再调用一个私有的generator来完成实际迭代工作。

用FooGenerator的例子说,假如max小于0是不符合要求的,那么应该这样写:
public static IEnumerable<int> FooGenerator( int max ) {
    if ( 0 > max ) {
        throw new ArgumentException( "Number must be non-negative", "max" );
    }
    return FooGeneratorCore( max );
}

private static IEnumerable<int> FooGenerator( int max ) {
    for ( var i = 0; i < max; ++i ) {
        yield return i;
    }
}
分享到:
评论

相关推荐

    Json转C#Model实体 JsonCSharpClassGenerator

    生成的C#类可以直接被`Newtonsoft.Json`库(或.NET Core中的`System.Text.Json`库)用于序列化和反序列化,实现JSON数据和C#对象之间的转换。 例如,假设有一个如下的JSON数据: ```json { "name": "John Doe", ...

    CodeGenerator:C#代码生成器,用于从数据库读取表结构并生成相应的代码

    【描述】:CodeGenerator是一款C#编写的实用工具,它的主要功能是从数据库中提取表结构信息,并根据这些信息自动生成对应的代码,极大地提高了开发效率,减少了手动编写重复代码的工作量。 【详细知识点】: 1. **...

    [转帖] 用C# Generator解决Hanoi塔问题

    在C#中,可以使用`yield return`关键字来实现Generator的返回。 以下是一个简化的C#代码示例,展示了如何使用Generator来解决Hanoi塔问题: ```csharp public static IEnumerable&lt;string&gt; Hanoi(int n, char from,...

    mybatis中的generator工具

    在本篇文章中,我们将深入探讨MyBatis Generator的使用方法、主要功能以及如何结合MyBatis框架来提升开发效率。 一、MyBatis Generator简介 MyBatis Generator是一个基于Java的代码生成器,它能够根据数据库表结构...

    MySQL数据库生成C#实体类工具

    本篇文章将详细介绍如何使用`FreeSql.Generator`这一工具来从MySQL数据库自动生成C#实体类。 #### 一、安装FreeSql.Generator 在开始使用`FreeSql.Generator`之前,首先需要将其安装到您的开发环境中。安装步骤...

    c#实体类生成工具 EntitysCodeGenerate

    这些表会被映射为C#类,每个字段对应类中的一个属性。 3. 生成配置:工具通常提供一些自定义选项,例如是否生成注释、是否忽略特定字段、属性的数据类型转换等,以满足不同开发者的编码风格和需求。 4. 输出结果:...

    Node.js-在Express中愉快地使用Generator

    - Generator中的错误会被自动抛出,可以在外层捕获。 - 示例: ```javascript app.get('/', function*(req, res) { try { var result = yield someAsyncTask(); res.send(result); } catch (err) { res....

    JavaScript 中使用 Generator的方法

    在JavaScript中,Generator是一种特殊的函数,它的主要特性是可以暂停和恢复执行。Generator函数的返回值是一个迭代器对象,使用yield关键字来暂停函数的执行,并返回一个值;再次调用迭代器的next()方法时,...

    mybatis-generator-core-1.3.7-API文档-中文版.zip

    赠送jar包:mybatis-generator-core-1.3.7.jar; 赠送原API文档:mybatis-generator-core-1.3.7-javadoc.jar; 赠送源代码:mybatis-generator-core-1.3.7-sources.jar; 赠送Maven依赖信息文件:mybatis-generator-...

    openapi-generator,openapi生成器允许自动生成api客户端库(sdk生成)、服务器存根、文档和配置,并给出openapi规范(v2,v3).zip

    OpenAPI Generator则是这个规范的实用工具,通过解析规范文件(通常是.yaml或.json格式),它可以生成多种编程语言的SDK(Software Development Kit),如Java、Python、C#、JavaScript等,这样开发者就可以直接在...

    LR-Virtual_User_Generator完全中文使用说明

    3. **脚本语言**:LoadRunner使用VU Generator(VuGen)中的内置脚本语言——Vuser Language(VuScript)。VuScript类似于C语言,允许用户自定义函数和变量,进行条件判断和循环控制。 4. **参数化**:为了模拟真实...

    Msc-generator使用手册

    Msc-generator是一款用于从文本描述中绘制各种图表的工具,它的版本为6.1.0,更新于2017年11月3日。这款工具可以生成多种类型的图表,包括信号图、流程图、块图等,每种图表都有其特定的用途和特点。为了更好地理解...

    generator-ajax.zip

    在这个例子中,`fetch`操作被`yield`暂停,外部可以通过`next`方法传递结果给Generator,使得Generator能够继续执行。 接下来,我们讨论js进程中断。在JavaScript中,由于其单线程特性,一旦开始执行,无法直接中断...

    mybatis-generator生成数据库中文注释

    首先,你需要在项目中引入MyBatis Generator的依赖。如果是Maven项目,可以在pom.xml中添加以下依赖: ```xml &lt;groupId&gt;org.mybatis.generator &lt;artifactId&gt;mybatis-generator-core &lt;version&gt;1.4.2 ``` 接...

    mybatis逆向工具generator,中文注释,Byte改Integer

    MyBatis Generator是一款强大的逆向工程工具,它能够根据数据库中的表自动生成对应的Java实体类、Mapper接口和XML配置文件,极大地提高了开发效率。在给定的标题中,“mybatis逆向工具generator,中文注释,Byte改...

    form-generator项目集成方案.doc

    在引入form-generator项目时,需要复制form-generator项目的components文件夹下的全部文件、uits文件夹下的所有文件以及views文件夹下全部文件到自己项目中,styles文件夹下全部文件复制到自己工程,icons文件夹下...

    mybatis-generator 解决中文注释乱码

    如果是eclipse工具,直接生成的,则替换:plugins/org.mybatis.generator.core_1.3.2*****.jar,如果是其他方式则直接替换掉官方的mybatis-generator-core1.3.2.jar即可

    C# ffmpeg 处理视频、C# ffmpeg播放视频

    FFmpeg库被许多编程语言所支持,包括C#。本篇将深入探讨如何在C#环境下利用FFmpeg进行视频处理以及播放视频。 首先,`C# ffmpeg.autogen`是指将FFmpeg的C语言接口自动生成为C#可用的接口,以便于在C#项目中调用...

    EF 4.x POCO Entity Generator for C#

    做个记号。自己收集使用。 你也可通过以下链接下载 https://visualstudiogallery.msdn.microsoft.com/23df0450-5677-4926-96cc-173d02752313/

Global site tag (gtag.js) - Google Analytics