C#使用yield關(guān)鍵字提升迭代性能與效率
前言
yield關(guān)鍵字在C#中簡(jiǎn)化了數(shù)據(jù)迭代的方式,實(shí)現(xiàn)了按需生成數(shù)據(jù),自動(dòng)維護(hù)迭代狀態(tài),減少了內(nèi)存占用,并允許在迭代時(shí)執(zhí)行復(fù)雜邏輯。
傳統(tǒng)迭代和yield迭代方式對(duì)比
咱們來(lái)看看傳統(tǒng)迭代方式和yield關(guān)鍵字迭代方式對(duì)比,是否如傳說(shuō)中的代碼實(shí)現(xiàn)起來(lái)更簡(jiǎn)潔和高效:
/// <summary>
/// 傳統(tǒng)迭代方式和yield關(guān)鍵字迭代方式對(duì)比
/// </summary>
public static void IteratorComparisonRun()
{
Console.WriteLine("迭代器方法使用yield關(guān)鍵字:");
foreach (var number in GetNumbersWithYield())
{
Console.WriteLine(number);
}
Console.WriteLine("傳統(tǒng)迭代方法返回一個(gè)List<int>");
var numbers = GetNumbersWithoutYield();
foreach (var number in numbers)
{
Console.WriteLine(number);
}
}
/// <summary>
/// 迭代器方法使用yield關(guān)鍵字
/// </summary>
/// <returns></returns>
public static IEnumerable<int> GetNumbersWithYield()
{
for (int i = 0; i < 6; i++)
{
yield return i;
}
}
/// <summary>
/// 傳統(tǒng)迭代方法返回一個(gè)List<int>
/// </summary>
/// <returns></returns>
public static List<int> GetNumbersWithoutYield()
{
var numbers = new List<int>();
for (int i = 0; i < 6; i++)
{
numbers.Add(i);
}
return numbers;
}
輸出結(jié)果:
yield延遲加載按需獲取數(shù)據(jù)
yield關(guān)鍵字可以通過(guò)延遲執(zhí)行的方式,僅在實(shí)際需要時(shí)生成數(shù)據(jù),從而提高了性能和效率。
/// <summary>
/// yield關(guān)鍵字延遲加載按需獲取數(shù)據(jù)
/// </summary>
public static void LazyLoadingRun()
{
Console.WriteLine("yield延遲加載按需獲取數(shù)據(jù) 開(kāi)始...");
foreach (var number in GetEvenNumbers(11))
{
Console.WriteLine($"返回值 === {number} ===");
Thread.Sleep(500);
}
Console.WriteLine("yield延遲加載按需獲取數(shù)據(jù) 結(jié)束...");
}
/// <summary>
/// 使用yield返回偶數(shù)的迭代器方法
/// </summary>
/// <returns></returns>
public static IEnumerable<int> GetEvenNumbers(int number)
{
for (int i = 1; i < number; i++)
{
Console.WriteLine($"Yielding {i}");
if (i % 2 == 0)
{
yield return i; //只在需要時(shí)生成偶數(shù)
}
}
}
輸出結(jié)果:
圖片
yield break顯式示迭代結(jié)束
yield break:顯式示迭代結(jié)束,如以下示例所示:
public static void YieldBreakRun()
{
Console.WriteLine(string.Join(" ", TakeWhilePositive(new int[] { 1, 3, 4, 5, -1, 3, 4 })));
//輸出:1 3 4 5
Console.WriteLine(string.Join(" ", TakeWhilePositive(new int[] { 9, 8, 7, 6, 5, -5, 88, 100 })));
//輸出:9 8 7 6 5
}
public static IEnumerable<int> TakeWhilePositive(IEnumerable<int> numbers)
{
foreach (int n in numbers)
{
if (n > 0)
{
yield return n;
}
else
{
yield break;
}
}
}
什么情況不能使用yield關(guān)鍵字
- 帶有 in、ref 或 out 參數(shù)的方法。
- Lambda 表達(dá)式和匿名方法。
- 在 C# 13 之前,yield 在具有 unsafe 塊的任何方法中都無(wú)效。從 C# 13 開(kāi)始,可以在包含 unsafe 塊的方法中使用 yield,但不能在 unsafe 塊中使用。
- 不能在catch和finally塊中使用yield return和yield break。
- 不能在具有catch塊的try塊中使用yield return和yield break。
- 可以在只有finally塊的try塊中使用yield return和yield break。
完整示例代碼
圖片