اندازهی قلم متن
تخمین مدت زمان مطالعهی مطلب:
سه دقیقه
عبارت foreach در زبان #C، امکان پیمایش اعضای یک مجموعه را میسر میکند؛ اما نه هر مجموعهای. این مجموعهی خاص باید به این صورت تعریف شده باشد:
مثال 1: نوع <IEnumerator<T با حلقهی foreach سازگار نیست
قابلیت پیمایش توسط حلقهی foreach را ندارد. اگر در C# 8.0 این حلقه را بر روی آن اعمال کنیم، به خطای کامپایلر زیر میرسیم:
اما میتوان به صورت زیر در C# 9.0، این متد را به آن اضافه کرد:
اکنون حلقهی foreach را میتوان بر روی نوعهای <IEnumerator<T نیز بکار گرفت:
این نکته بر روی نمونهی async آن نیز قابل اعمال است که مثالی از آنرا در ادامه مشاهده میکنید:
مثال زیر را درنظر بگیرید:
در اینجا سعی کردهایم تا حلقهی foreach را بر روی یک tuple سه عضوی، اعمال کنیم. اما با خطای کامپایلر زیر مواجه میشویم:
برای رفع این خطا در C# 9.0 تنها کافی است متد الحاقی GetEnumerator مخصوص نوع آنرا طراحی و به برنامه اضافه کرد:
الف) <IEnumerable<T را پیاده سازی کرده باشد.
ب) و یا ... مهم نیست که این مجموعه حتما <IEnumerable<T را پیاده سازی کرده باشد. اگر این مجموعه به همراه یک متد عمومی خاص با نام GetEnumerator باشد که خروجی آن دارای خاصیت عمومی T Current است (یکی از اعضای اینترفیس <IEnumerable<T) و همچنین به همراه متد عمومی bool MoveNext نیز هست (یکی از اعضای اینترفیس IEnumerator)، قابلیت کار با حلقهی foreach را پیدا میکند و ... اکنون در C# 9.0 میتوان متد GetEnumerator را به صورت یک متد الحاقی، به هر نوع دلخواهی اعمال کرد! یعنی میتوان برای هر نوعی در صورت نیاز، یک GetEnumerator خاص را طراحی کرد که سبب به کار افتادن حلقهی foreach بر روی آن شود.
مثال 1: نوع <IEnumerator<T با حلقهی foreach سازگار نیست
نوع <IEnumerator<T به دلیل نداشتن متد عمومی GetEnumerator که ذکر شد:
public interface IEnumerator<out T> : IEnumerator, IDisposable { // // Summary: // Gets the element in the collection at the current position of the enumerator. // // Returns: // The element in the collection at the current position of the enumerator. T Current { get; } }
Error CS1579 foreach statement cannot operate on variables of type ‘IEnumerator’ because ‘IEnumerator’ does not contain a public instance or extension definition for ‘GetEnumerator’
static class Extensions { public static IEnumerator<T> GetEnumerator<T>(this IEnumerator<T> enumerator) => enumerator; }
اکنون حلقهی foreach را میتوان بر روی نوعهای <IEnumerator<T نیز بکار گرفت:
class Program { void Main() { var enumerator = Enumerable.Range(0, 10).GetEnumerator(); foreach (var item in enumerator) { Console.WriteLine(item); } } }
این نکته بر روی نمونهی async آن نیز قابل اعمال است که مثالی از آنرا در ادامه مشاهده میکنید:
static class Extensions { public static IAsyncEnumerator<T> GetAsyncEnumerator<T>(this IAsyncEnumerator<T> enumerator) => enumerator; } class Program { static async Task Main() { var enumerator = GetAsyncEnumerator(); await foreach (var item in enumerator) { Console.WriteLine(item); } } static async IAsyncEnumerator<int> GetAsyncEnumerator() { yield return 0; await Task.Delay(1); yield return 1; } }
مثال 2: اضافه کردن پشتیبانی از حلقهی foreach بر روی نوعهای tuple
مثال زیر را درنظر بگیرید:
class Program { static void Main() { foreach (var item in (1, 2, 3)) { Console.WriteLine(item); } } }
foreach statement cannot operate on variables of type '(int, int, int)' because '(int, int, int)' does not contain a public instance or extension definition for 'GetEnumerator' [CS9Features]csharp(CS1579)
static class Extensions { public static IEnumerator<object> GetEnumerator<T1, T2, T3>(this ValueTuple<T1, T2, T3> tuple) { yield return tuple.Item1; yield return tuple.Item2; } }