更新時間:2024-12-31 09:56:10作者:佚名
我們學(xué)習(xí)知識的目的就是為什么要使用yield關(guān)鍵字,不使用可以嗎?相信會更有趣。
首先貼出正常輸出偶數(shù)集的方法。
/*
*
* 學(xué)習(xí)Yield Return 語法
* 使用兩個方法,顯示1 - 100之間的全部偶數(shù)
*
*
*/
using System;
using System.Collections.Generic;
namespace YieldReturn語法解析
{
class Program
{
static private List<int> _numArray; //用來保存1-100 這100個整數(shù)
Program() //構(gòu)造函數(shù)。我們可以通過這個構(gòu)造函數(shù)往待測試集合中存入1-100這100個測試數(shù)據(jù)
{
_numArray = new List<int>(); //給集合變量開始在堆內(nèi)存上開內(nèi)存,并且把內(nèi)存首地址交給這個_numArray變量
for (int i = 1; i <= 100; i++)
{
_numArray.Add(i); //把1到100保存在集合當(dāng)中方便操作
}
}
static void Main(string[] args)
{
new Program();
TestMethod();
}
//測試求1到100之間的全部偶數(shù)
static public void TestMethod()
{
foreach (var item in GetAllEvenNumber())
{
Console.WriteLine(item); //輸出偶數(shù)測試
}
}
//測試我們正常情況下拿到全部偶數(shù)的方法
static IEnumerable<int> GetAllEvenNumber()
{
List<int> result = new List<int>(); //開集合內(nèi)存存偶數(shù)用
foreach (int num in _numArray)
{
if(num % 2 == 0) //判斷是不是偶數(shù)
{
//yield return num;
result.Add(num); //存入集合
}
}
//返回偶數(shù)集合變量 可能有人會覺得奇怪返回類型不是List這樣可以嗎
//這個就要回到我們的里氏替換原則了,子類是可以替換父類的,也就是當(dāng)父類用
//比如我這個方法是想得到IEnumerable 類型變量,但是我給了List類型變量
//注意List 是繼承 IEnumerable 的,什么意思當(dāng)我們把子類當(dāng)父類使用,
//那么大才小用,因?yàn)樽宇惡芏喽际抢^承父親,你自身增加很多字段或者方法,這樣就不能用了。
return result;
//yield break;
}
}
}
發(fā)布使用 Yiele Return 方法輸出耦合集合
/*
*
* 學(xué)習(xí)Yield Return 語法
* 使用兩個方法,顯示1 - 100之間的全部偶數(shù)
*
*
*/
using System;
using System.Collections.Generic;
namespace YieldReturn語法解析
{
class Program
{
static private List<int> _numArray; //用來保存1-100 這100個整數(shù)
Program() //構(gòu)造函數(shù)。我們可以通過這個構(gòu)造函數(shù)往待測試集合中存入1-100這100個測試數(shù)據(jù)
{
_numArray = new List<int>(); //給集合變量開始在堆內(nèi)存上開內(nèi)存,并且把內(nèi)存首地址交給這個_numArray變量
for (int i = 1; i <= 100; i++)
{
_numArray.Add(i); //把1到100保存在集合當(dāng)中方便操作
}
}
static void Main(string[] args)
{
new Program();
TestMethod();
}
//測試求1到100之間的全部偶數(shù)
static public void TestMethod()
{
foreach (var item in GetAllEvenNumber())
{
Console.WriteLine(item); //輸出偶數(shù)測試
}
}
//測試我們使用Yield Return情況下拿到全部偶數(shù)的方法
static IEnumerable<int> GetAllEvenNumber()
{
foreach (int num in _numArray)
{
if(num % 2 == 0) //判斷是不是偶數(shù)
{
yield return num; //返回當(dāng)前偶數(shù)
}
}
yield break; //當(dāng)前集合已經(jīng)遍歷完畢,我們就跳出當(dāng)前函數(shù),其實(shí)你不加也可以
//這個作用就是提前結(jié)束當(dāng)前函數(shù),就是說這個函數(shù)運(yùn)行完畢了。
}
}
}
你測試過這2個代碼的結(jié)果嗎?你能正確得到所有偶數(shù)組嗎?我需要你專門測試一下。只有這樣,你才能快速進(jìn)步,才能真正學(xué)會。只要觀看并練習(xí)假動作即可。
現(xiàn)在我們來談?wù)勊鼈兊膮^(qū)別:
這才是你真正需要學(xué)習(xí)的地方
我們需要設(shè)置一個斷點(diǎn)
我希望您在斷點(diǎn)處進(jìn)行調(diào)試。如果你貼的細(xì)節(jié)太多,你會發(fā)現(xiàn)in return是什么意思,當(dāng)代碼到達(dá)Yield return num時,當(dāng)前函數(shù)就會結(jié)束,num會被移交給這里。我先給你看圖片。
最后,將給出 item 變量。
然后輸出,然后執(zhí)行GetAllEvenNumber()方法,與最后一個偶數(shù)相同。
我們發(fā)現(xiàn)這個Yield Return可以將當(dāng)前函數(shù)的進(jìn)程狀態(tài)切換為阻塞狀態(tài)貝語網(wǎng)校,然后選擇將CPU交給當(dāng)前傳出進(jìn)程,從而可以轉(zhuǎn)而執(zhí)行調(diào)用者函數(shù)。 (補(bǔ)充一點(diǎn)知識,我們寫的程序添加到內(nèi)存中并不一定是一個進(jìn)程,我們會根據(jù)情況將其劃分為若干個子進(jìn)程,以方便操作系統(tǒng)的管理以及多個進(jìn)程的運(yùn)行)內(nèi)存中的程序以提高計(jì)算機(jī)資源的利用率)
這有一個優(yōu)點(diǎn)。如果我們有 1,000,000 個數(shù)據(jù),我們需要獲取內(nèi)部的耦合。當(dāng)我們通過這個方法得到一個耦合后in return是什么意思,它會立即顯示在控制臺上。不用等待很長時間,而是找到所有偶數(shù)并將其存儲在集合中,然后一一輸出。
這個好處是巨大的。比如我們的用戶可能只是一開始看數(shù)據(jù),確定不是所有的值都是需要的,讀完這些之后再看剩下的,這樣數(shù)據(jù)就會感覺顯示沒有延遲。
接下來,我們斷點(diǎn)并調(diào)整通用函數(shù)來查找偶數(shù)。
我們發(fā)現(xiàn)這個循環(huán)在完成之前不會退出當(dāng)前函數(shù),也就是說,我們必須找到所有偶數(shù)。所以如果我們有很多數(shù)據(jù)。它將等待該數(shù)據(jù),然后才能獲取該數(shù)據(jù)并將其顯示給用戶。
這就麻煩了。
所以我們得出結(jié)論:Yield Return關(guān)鍵字的作用是退出當(dāng)前函數(shù),保存當(dāng)前函數(shù)執(zhí)行的地方,也就是上下文。你發(fā)現(xiàn)上次運(yùn)行的代碼不會再執(zhí)行,直到下次執(zhí)行這個函數(shù)。
但你一般的返回結(jié)果是,如果你在循環(huán)體內(nèi)提前返回,下面調(diào)用這個函數(shù)就會從第一步重新執(zhí)行。不記錄最后執(zhí)行位置。