از زمان ارائهی C# 5 و معرفی الگوهای async/await، تنها نوعهای خروجی پشتیبانی شده، <Task، Task<T و void (در موارد خاص) بودند. مشکل همراه با این روش، اجبار به وهله سازی رسمی یک Task است؛ حتی اگر نوع خروجی کاملا مشخص باشد.
برای نمونه در متد ذیل، میزان حجم مصرفی در یک پوشه بازگشت داده میشود:
public async Task<long> GetDirectorySize(string path, string searchPattern)
{
if (!Directory.EnumerateFileSystemEntries(path, searchPattern).Any())
return 0;
else
return await Task.Run<long>(() => Directory.GetFiles(path, searchPattern,
SearchOption.AllDirectories).Sum(t => (new FileInfo(t).Length)));
}
اگر پوشهای خالی باشد، حجم آن صفر است و در این حالت نیازی به ایجاد یک ترد مخصوص آن نیست. اما با توجه به اینکه خروجی متد، <Task<long است، هنوز هم باید این Task وهله سازی شود. برای نمونه اگر به کدهای IL آن دقت کنیم، return 0 آن به صورت ذیل ترجمه میشود:
AsyncTaskMethodBuilder<long>.Create()
باید دقت داشت که Task، یک نوع ارجاعی است و استفادهی از آن به معنای تخصیص حافظهاست. اما زمانیکه قسمتی از کد کاملا همزمان اجرا میشود و یا مقداری کش شده را بازگشت میدهد، این تخصیص حافظهی اضافی، خصوصا اگر در حلقهها بکار گرفته شود، هزینهبر خواهد بود.
امکان تعریف خروجیهای سفارشی متدهای async در C# 7.0
در C# 7 میتوان خروجیهای سفارشی را جهت متدهای async تعریف کرد و پیشنیاز اصلی آن پیاده سازی متد GetAwater است. برای مثال <System.Threading.Tasks.ValueTask<T یک چنین نوع سفارشی را ارائه میدهد. در این حالت، متد ابتدای بحث را میتوان به صورت ذیل بازنویسی کرد:
public async ValueTask<long> GetDirectorySize(string path, string searchPattern)
{
if (!Directory.EnumerateFileSystemEntries(path, searchPattern).Any())
return 0;
else
return await Task.Run<long>(() => Directory.GetFiles(path, searchPattern,
SearchOption.AllDirectories).Sum(t => (new FileInfo(t).Length)));
}
اگر دقت کنید بجز تغییر نوع خروجی متد، تغییر دیگری نیاز نبودهاست.
همانطور که از نام ValueTask نیز مشخص است، یک struct است؛ برخلاف Task و تخصیص حافظهی آن بر روی stack بجای heap صورت میگیرد. به این ترتیب با کاهش فشار بر روی GC، در حلقههایی که خروجی value type دارند، با اندازه گیریهای انجام شده، کارآیی تا 50 درصد هم میتواند بهبود یابد.
برای کامپایل قطعه کد فوق و تامین نوع جدید ValueTask، نیاز به نصب بستهی نیوگت ذیل نیز میباشد:
PM> install-package System.Threading.Tasks.Extensions