天天看點

yield語句

yield簡介

yield是自C#第二個版本一來所添加的新的文法,目的是為了簡化foreach疊代集合。第一個版本疊代集合需要建立疊代器Enumerator

yield的好處:

為了簡化foreach疊代集合是建立疊代器。

foreach疊代(手動建立疊代器)

那麼不用yield來簡化建立疊代器,需要相當繁瑣的代碼

  1. 首先需要疊代的類必須繼承 IEnumerable 接口并且實作接口的成員 GetEnumerator 方法
  2. 其次 GetEnumerator 方法傳回 IEnumerator 類型的對象
  3. 則需要建立一個疊代器繼承自IEnumerator接口一句IDisposable接口,并實作接口中的方法。

代碼示例如下:

using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConsoleApplication26
{
    class Program
    {
        static void Main(string[] args)
        {
            HelloList list = new HelloList();
            foreach (var item in list)
            {
                Console.WriteLine(item);
            }
        }
    }
    public class HelloList : IEnumerable
    {

        public IEnumerator GetEnumerator()
        {
            Enumerator e = new Enumerator();
            return e;
        }
    }
    public class Enumerator : IEnumerator, IDisposable
    {
        private int state;
        private string current;
        public Enumerator(int state)
        {
            this.state = state;
        }

        public void Dispose()
        {
        }

        public object Current
        {
            get { return current; }
        }

        public bool MoveNext()
        {
            switch (state)
            {
                case :
                    current = "Hello";
                    state = ;
                    return true;
                case :
                    current = "world";
                    state = ;
                    return true;
                default:
                    return false;
            }
        }

        public void Reset()
        {
            throw new NotImplementedException();
        }
    }

}
           

從中可以看出,首先我們建立一個集合類 HelloList 繼承 IEnumerable 接口并且實作了 GetEnumerator 方法,

然後建立了一個疊代器 Enumerator 類 繼承 IEnumerator 接口 并且實作裡面的方法 MoveNext 屬性 Current 等。過程相當繁瑣。

foreach疊代(yield自動建立疊代器)

通過如下代碼和上述手動建立疊代器的代碼做對比,将會發現yield簡直是太好用了,簡化了很多代碼的編寫。

using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConsoleApplication26
{
    class Program
    {
        static void Main(string[] args)
        {
            HelloList list = new HelloList();
            foreach (var item in list)
            {
                Console.WriteLine(item);
            }
        }
    }
    public class HelloList : IEnumerable
    {

        public IEnumerator GetEnumerator()
        {
            yield return "Hello";
            yield return "World";
        }
    }

}
           

yield文法介紹

通過上面兩節,我們已經知道了使用yield的好處,那麼yield具體有什麼文法,已經文法上的一些限制是什麼呢??

文法:

yield return

傳回集合的一個元素,并移動到下一個元素上。如上,傳回Hello并且移動到World語句上。

yield break

終止疊代,很簡單的例子,比如,有10個元素,我想疊代到8個元素時終止疊代
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConsoleApplication26
{
    class Program
    {
        static void Main(string[] args)
        {
            HelloList list = new HelloList();
            foreach (var item in list)
            {
                Console.WriteLine(item);
            }
        }
    }
    public class HelloList : IEnumerable
    {
        public IEnumerator GetEnumerator()
        {
            for (int i = ; i <= ; i++)
            {
                if (i > ) yield break;
                yield return i;
            }
        }
    }

}
           

文法已經弄清楚了,那麼使用yield有什麼限制呢??

  1. 包含yield語句的方法和屬性都是疊代快
  2. 疊代快必須傳回IEnumerator或者IEnumerable類型或者這些版本的泛型類型
  3. 疊代快可以包含多條yield return和yield break,但是不能包含 return。