zoukankan      html  css  js  c++  java
  • yield个人理解及简明示例

    1.写法有2种:
    yield return <expression>和yield break
    yield用于在迭代中返回一个值,并将值带入下一次迭代中。yield break则意味着停止迭代。纯粹的文字描述,一千个人有一千个说法,还是用代码更容易说清楚。
    2.官方示例(略带修改):

     private void button1_Click(object sender, EventArgs e)
            {
                string s = string.Empty;
                foreach (int i in List.Power(2,8))
                {
                    s += i.ToString() + ",";
                }
                MessageBox.Show(s);
            }
        public class List
        {
            //using System.Collections;
            public static IEnumerable Power(int number, int exponent)
            {
                int counter = 0;
                int result = 1;
                
                while (counter++ < exponent)
                {
                    result = result * number;
                    yield return result;
                }
            }
        }
    

    运行示例,发现power方法中的while代码部分,每循环执行一次,即输出一个值,并将这个值带入下一次循环,而power函数并没有每次被调用。

    为了验证,我们修改下官方示例代码,来看看我们的判断是否有误:

    private void button1_Click(object sender, EventArgs e)
            {
                string s = string.Empty;
                foreach (int i in List.Power())
                {
                    s += i.ToString() + ",";
                }
                MessageBox.Show(s);
            }
        public static IEnumerable Power()
            {
                int counter = 0;
                int result = 1;
                int number = 2, exponent = 8;
                while (counter++ < exponent)
                {
                    result = result * number;
                    yield return result;
                }
            }
    

     运行结果与官方示例相同,说明.net framework每次只把yield部分所在的部分代码进行了迭代返回处理。

    3.官方的另一个示例为用yield作为属性(输出方式略有修改)。

     private void button2_Click(object sender, EventArgs e)
            {
                var theGalaxies = new Galaxies();
                string ps = string.Empty;
                foreach (Galaxy theGalaxy in theGalaxies.NextGalaxy)
                {
                    ps += (theGalaxy.Name + " " + theGalaxy.MegaLightYears.ToString() + " >> ");
                }
                MessageBox.Show(ps);
            } 
    public System.Collections.Generic.IEnumerable<Galaxy> NextGalaxy
            {
                get
                {
                    yield return new Galaxy { Name = "Tadpole", MegaLightYears = 400 };
                    yield return new Galaxy { Name = "Pinwheel", MegaLightYears = 25 };
                    yield return new Galaxy { Name = "Milky Way", MegaLightYears = 0 };
                    yield return new Galaxy { Name = "Andromeda", MegaLightYears = 3 };
                }
    
            }
    

     输出结果:

    从这个例子可以看出yield其实就是临时中断执行,输出后继续执行而已。

    可以修改下这个例子,看效果如何:

     public class Galaxies
        {
            List<Galaxy> ls = new List<Galaxy>();
            public System.Collections.Generic.IEnumerable<Galaxy> NextGalaxy
            {
                get
                {
                    yield return new Galaxy { Name = "Tadpole", MegaLightYears = 400 };
                    yield return new Galaxy { Name = "Pinwheel", MegaLightYears = 25 };
                    yield return new Galaxy { Name = "Milky Way", MegaLightYears = 0 };
                    yield return new Galaxy { Name = "Andromeda", MegaLightYears = 3 };
                }
    
            }
            public System.Collections.Generic.IEnumerable<Galaxy> NextGalaxy1
            {
    
                get
                {
                    ls.Add(new Galaxy { Name = "Tadpole", MegaLightYears = 400 });
                    ls.Add(new Galaxy { Name = "Pinwheel", MegaLightYears = 25 });
                    ls.Add(new Galaxy { Name = "Milky Way", MegaLightYears = 0 });
                    ls.Add(new Galaxy { Name = "Andromeda", MegaLightYears = 3 });
                    int i = -1;
                    while (i++ < ls.Count - 1)
                    {
                        yield return ls[i];
                    }
                }
            }
        }
    

     调用NextGalaxy1后,结果与官方示例结果相同,还可以进一步修改NextGalaxy1,使其更容易别理解:

     public System.Collections.Generic.IEnumerable<Galaxy> NextGalaxy1
            {
    
                get
                {
                    ls.Add(new Galaxy { Name = "Tadpole", MegaLightYears = 400 });
                    ls.Add(new Galaxy { Name = "Pinwheel", MegaLightYears = 25 });
                    ls.Add(new Galaxy { Name = "Milky Way", MegaLightYears = 0 });
                    ls.Add(new Galaxy { Name = "Andromeda", MegaLightYears = 3 });
                    int i = 0;
                    while (i < ls.Count)
                    {
                        yield return ls[i];
                        i++;
                    }
                }
            }
    

     这样看来就很容易理解其含义了,更进一步说就是一边给你输出结果,一边继续给你执行代码,一举两得!

    如果想中断执行,则直接用yield break即可。

    代码如下:

     public System.Collections.Generic.IEnumerable<Galaxy> NextGalaxy1
            {
    
                get
                {
                    ls.Add(new Galaxy { Name = "Tadpole", MegaLightYears = 400 });
                    ls.Add(new Galaxy { Name = "Pinwheel", MegaLightYears = 25 });
                    ls.Add(new Galaxy { Name = "Milky Way", MegaLightYears = 0 });
                    ls.Add(new Galaxy { Name = "Andromeda", MegaLightYears = 3 });
                    int i = -1;
                    while (i++ < ls.Count - 1)
                    {
                        yield return ls[i];
                        if (ls[i].MegaLightYears == 0)
                        {
                            yield break;
                        }
    
                    }
                }
            }
    

     输出结果为:

    附官方链接:http://msdn.microsoft.com/zh-cn/library/9k7k7cf0.aspx

  • 相关阅读:
    抖动代码
    WSAWaitForMultipleEvents()
    udp 不需要 listen
    WSAEventSelect
    C++ Win32控制台应用程序捕捉关闭事件
    玩转Win32开发(2):完整的开发流程
    win32线程池代码(WinApi/C++)
    又线程类封装
    Win2 Socket(套接字)相关 API
    Win32函数Sleep的精度测试
  • 原文地址:https://www.cnblogs.com/jizhong/p/3809662.html
Copyright © 2011-2022 走看看