{"id":1226,"date":"2022-05-15T16:47:06","date_gmt":"2022-05-15T08:47:06","guid":{"rendered":"https:\/\/summid.icu\/?p=1226"},"modified":"2022-06-20T22:38:19","modified_gmt":"2022-06-20T14:38:19","slug":"extending-the-async-methods-in-c","status":"publish","type":"post","link":"https:\/\/summid.icu\/index.php\/2022\/05\/15\/extending-the-async-methods-in-c\/","title":{"rendered":"\u62d3\u5c55C#\u7684\u5f02\u6b65\u65b9\u6cd5"},"content":{"rendered":"<p>\u672c\u6587\u624b\u52a8\u7ffb\u8bd1\u81ea\uff1a<a href=\"https:\/\/devblogs.microsoft.com\/premier-developer\/extending-the-async-methods-in-c\/\" target=\"_blank\" rel=\"noopener\">Extending the async methods in C#<\/a><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u4e0a\u4e00\u7bc7\u6587\u7ae0\u4e2d\uff0c\u6211\u4eec\u8ba8\u8bba\u4e86 C# \u7f16\u8bd1\u5668\u662f\u5982\u4f55\u5904\u7406\u5f02\u6b65\u65b9\u6cd5\u7684\u3002\u5728\u8fd9\u4e00\u7bc7\uff0c\u6211\u4eec\u4f1a\u91cd\u70b9\u5173\u6ce8 C# \u7f16\u8bd1\u5668\u4e3a\u81ea\u5b9a\u4e49\u5f02\u6b65\u65b9\u6cd5\u63d0\u4f9b\u7684\u53ef\u6269\u5c55\u6027\u3002<\/span><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u6211\u4eec\u6709\u4e09\u79cd\u63a7\u5236\u5f02\u6b65\u65b9\u6cd5\u72b6\u6001\u673a\u7684\u601d\u8def\uff1a<\/span><\/p>\n<ol class=\"ol-list\" start=\"\">\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u5728 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>System.Runtime.CompilerServices<\/code><\/span><span class=\"md-plain\"> \u547d\u540d\u7a7a\u95f4\u4e2d\u63d0\u4f9b\u81ea\u5df1\u7684\u5f02\u6b65\u65b9\u6cd5 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>builder<\/code><\/span><span class=\"md-plain\"> \u3002<\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u81ea\u5b9a\u4e49 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task awaiters<\/code><\/span><span class=\"md-plain\"> \u3002<\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u81ea\u5b9a\u4e49 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain\"> \u7c7b\u3002<\/span><\/p>\n<\/li>\n<\/ol>\n<h2 class=\"md-end-block md-heading\"><span class=\"md-plain\">\u5728 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>System.Runtime.ComplierServices<\/code><\/span><span class=\"md-plain\"> \u547d\u540d\u7a7a\u95f4\u4e0b\u81ea\u5b9a\u4e49 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncMethodBuilder<\/code><\/span><span class=\"md-plain\"> \u7c7b<\/span><\/h2>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u4ece\u524d\u6587\u6211\u4eec\u5f97\u77e5\uff0cC# \u7f16\u8bd1\u5668\u4f1a\u4f9d\u636e\u4e00\u4e9b\u7cfb\u7edf\u63d0\u4f9b\u7684\u7c7b\u5c06\u5f02\u6b65\u65b9\u6cd5\u8f6c\u6362\u4e3a\u72b6\u6001\u673a\u3002\u4f46 C# \u7f16\u8bd1\u5668\u5e76\u4e0d\u8981\u6c42\u6240\u4f9d\u636e\u7684\u7c7b\u6765\u81ea\u6307\u5b9a\u7684\u67d0\u4e2a\u7a0b\u5e8f\u96c6\u3002\u4f8b\u5982\uff0c\u6211\u4eec\u53ef\u4ee5\u5728\u9879\u76ee\u4e2d\u63d0\u4f9b\u81ea\u5df1\u7684 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"http:\/\/referencesource.microsoft.com\/#mscorlib\/system\/runtime\/compilerservices\/AsyncMethodBuilder.cs,b07562c618ee846c\"><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncVoidMethodBuilder<\/code><\/span><\/a><\/span><span class=\"md-plain\"> \uff0c\u8fd9\u6837 C# \u7f16\u8bd1\u5668\u4f1a\u5c06\u5f02\u6b65\u673a\u5236\u4e0e\u8be5\u7c7b\u201c\u7ed1\u5b9a\u201d\u3002<\/span><\/p>\n<p class=\"md-end-block md-p md-focus\"><span class=\"md-plain\">\u4e0b\u9762\u7684\u4f8b\u5b50\u5c06\u5e2e\u52a9\u6211\u4eec\u7406\u89e3\u5e95\u5c42\u8f6c\u6362\uff0c\u5e76\u770b\u770b\u4ee3\u7801\u8fd0\u884c\u65f6\u53d1\u751f\u7684\u4e8b\u60c5\uff1a<\/span><\/p>\n\n\n<pre class=\"wp-block-code\"><code>namespace System.Runtime.CompilerServices\r\n{\r\n    \/\/ AsyncVoidMethodBuilder.cs in your project\r\n    public class AsyncVoidMethodBuilder\r\n    {\r\n        public AsyncVoidMethodBuilder()\r\n            => Console.WriteLine(\".ctor\");\r\n \r\n        public static AsyncVoidMethodBuilder Create()\r\n            => new AsyncVoidMethodBuilder();\r\n \r\n        public void SetResult() => Console.WriteLine(\"SetResult\");\r\n \r\n        public void Start&lt;TStateMachine>(ref TStateMachine stateMachine)\r\n            where TStateMachine : IAsyncStateMachine\r\n        {\r\n            Console.WriteLine(\"Start\");\r\n            stateMachine.MoveNext();\r\n        }\r\n \r\n        \/\/ AwaitOnCompleted, AwaitUnsafeOnCompleted, SetException \r\n        \/\/ and SetStateMachine are empty\r\n    }   \r\n}<\/code><\/pre>\n\n\n<p><span class=\"md-plain md-expand\">\u73b0\u5728\uff0c\u9879\u76ee\u4e2d\u7684\u6bcf\u4e2a\u5f02\u6b65\u65b9\u6cd5\u4f1a\u91c7\u7528\u81ea\u5b9a\u4e49\u7248\u672c\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncVoidMethodBuilder<\/code><\/span><span class=\"md-plain md-expand\"> \u3002\u6211\u4eec\u53ef\u4ee5\u7b80\u5355\u6d4b\u8bd5\u4e00\u4e0b\uff1a<\/span><\/p>\n\n\n<pre class=\"wp-block-code\"><code>&#91;Test]\r\npublic void RunAsyncVoid()\r\n{\r\n    Console.WriteLine(\"Before VoidAsync\");\r\n    VoidAsync();\r\n    Console.WriteLine(\"After VoidAsync\");\r\n \r\n    async void VoidAsync() { }\r\n}<\/code><\/pre>\n\n\n<p>\u8be5\u6d4b\u8bd5\u7684\u8f93\u51fa\uff1a<\/p>\n\n\n<pre class=\"wp-block-code\"><code>Before VoidAsync\r\n.ctor\r\nStart\r\nSetResult\r\nAfter VoidAsync<\/code><\/pre>\n\n\n<p class=\"md-end-block md-p\"><span class=\"md-plain md-expand\">\u4f60\u8fd8\u53ef\u4ee5\u5b9e\u73b0 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>UnsafeAwaitOnComplete<\/code><\/span><span class=\"md-plain\"> \u65b9\u6cd5\uff0c\u6765\u8bd5\u9a8c\u5f53\u5f02\u6b65\u65b9\u6cd5\u8fd4\u56de\u8fd8\u6ca1\u5b8c\u6210\u4efb\u52a1\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>Task<\/code><\/span><span class=\"md-plain\"> \u5bf9\u8c61\u7684\u65f6\u5019\u7684\u60c5\u51b5\u3002\u5b8c\u6574\u7684\u4f8b\u5b50\u53ef\u4ee5\u5728 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/github.com\/SergeyTeplyakov\/EduAsync\/blob\/master\/src\/01_AsyncVoidBuilder\/\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">github<\/span><\/a><\/span><span class=\"md-plain\"> \u627e\u5230\u3002<\/span><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u8981\u4fee\u6539 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>async Task<\/code><\/span><span class=\"md-plain\"> \u548c <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>async Task&lt;T&gt;<\/code><\/span><span class=\"md-plain\"> \u65b9\u6cd5\u7684\u884c\u4e3a\uff0c\u4f60\u9700\u8981\u63d0\u4f9b\u81ea\u5b9a\u4e49\u7248\u672c\u7684 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"http:\/\/referencesource.microsoft.com\/#mscorlib\/system\/runtime\/compilerservices\/AsyncMethodBuilder.cs,c983aa3f7c40052f\"><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncTaskMethodBuilder<\/code><\/span><\/a><\/span><span class=\"md-plain\"> \u548c <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"http:\/\/referencesource.microsoft.com\/#mscorlib\/system\/runtime\/compilerservices\/AsyncMethodBuilder.cs,5916df9e324fc0a1\"><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncTaskMethodBuilder<\/code><\/span><\/a><\/span><span class=\"md-plain\"> \u3002<\/span><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u5173\u4e8e\u4e0a\u9762\u63d0\u5230\u7684\u7c7b\uff0c\u53ef\u4ee5\u5728 github \u4e0a\u7684 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/github.com\/SergeyTeplyakov\/EduAsync\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">EduAsync<\/span><\/a><\/span><span class=\"md-plain\"> \u9879\u76ee\u91cc\u627e\u5230\u53c2\u8003\uff1a <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/github.com\/SergeyTeplyakov\/EduAsync\/blob\/master\/src\/02_AsyncTaskBuilder\/AsyncTaskMethodBuilder.cs\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">AsyncTaskBuilder.cs<\/span><\/a><\/span><span class=\"md-plain\"> \u548c <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/github.com\/SergeyTeplyakov\/EduAsync\/blob\/master\/src\/03_AsyncTaskBuilderOfT\/AsyncTaskMethodBuilderOfT.cs\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">AsyncTaskMethodBuilderOfT.cs<\/span><\/a><\/span><span class=\"md-plain\"> \u3002<\/span><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u611f\u8c22 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/codeblog.jonskeet.uk\/category\/eduasync\/\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">Jon Skeet<\/span><\/a><\/span><span class=\"md-plain\"> \u5728\u8fd9\u9879\u76ee\u4e0a\u7ed9\u4e88\u6211\uff08\u4f5c\u8005\uff09\u7684\u7075\u611f\u3002<\/span><\/p>\n<h2 class=\"md-end-block md-heading\"><span class=\"md-plain\">\u81ea\u5b9a\u4e49 awaiters<\/span><\/h2>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u524d\u9762\u7684\u4f8b\u5b50\u6709\u4ebf\u70b9 &#8220;\u9ed1\u9b54\u6cd5&#8221; \u7684\u5473\u9053\uff0c\u5e76\u4e14\u5b83\u4e0d\u9002\u5408\u7528\u5728\u771f\u6b63\u7684\u751f\u4ea7\u73af\u5883\u91cc\u3002\u6211\u4eec\u53ef\u4ee5\u901a\u8fc7\u5b83\u5b66\u4e60\u5f02\u6b65\u673a\u5236\uff0c\u4f46\u4f60\u80af\u5b9a\u4e0d\u60f3\u770b\u5230\u8fd9\u4e9b\u4ee3\u7801\u51fa\u73b0\u5728\u81ea\u5df1\u7684\u9879\u76ee\u91cc\u3002\u597d\u5728 C# \u4f5c\u8005\u7ed9\u7f16\u8bd1\u5668\u5185\u7f6e\u4e86\u53ef\u6269\u5c55\u7684\u70b9\uff0c\u5141\u8bb8\u5728\u5f02\u6b65\u65b9\u6cd5\u4e2d <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>await<\/code><\/span><span class=\"md-plain\"> \u4e0d\u540c\u7684\u5bf9\u8c61\u7c7b\u578b\u3002<\/span><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u53ef\u88ab <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>await<\/code><\/span><span class=\"md-plain\"> \u7684\u7c7b\uff08\u5373\u5728 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>await<\/code><\/span><span class=\"md-plain\"> \u8868\u8fbe\u5f0f\u4e2d\u5408\u6cd5\uff09\u9700\u8981\u6ee1\u8db3\u4ee5\u4e0b\u51e0\u4e2a\u6761\u4ef6\uff1a<\/span><\/p>\n<ul class=\"ul-list\" data-mark=\"-\">\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u8be5\u7c7b\u4e2d\u9700\u8981\u6709\u540d\u4e3a <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>GetAwaiter<\/code><\/span><span class=\"md-plain\"> \u7684\u65b9\u6cd5\uff08\u6269\u5c55\u65b9\u6cd5\u4e5f\u53ef\u4ee5\uff09\uff0c\u5426\u5219\u7f16\u8bd1\u5668\u4f1a\u62a5\u9519\u3002\u8be5\u65b9\u6cd5\u7684\u8fd4\u56de\u503c\u4e5f\u6709\u4e24\u5904\u9650\u5236\uff1a<\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u88ab\u8fd4\u56de\u7684\u5bf9\u8c61\u9700\u8981\u5b9e\u73b0 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"http:\/\/referencesource.microsoft.com\/#mscorlib\/system\/runtime\/compilerservices\/INotifyCompletion.cs,23\"><span class=\"md-pair-s\" spellcheck=\"false\"><code>INotifyCompletion<\/code><\/span><\/a><\/span><span class=\"md-plain\"> \u63a5\u53e3\u3002<\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u88ab\u8fd4\u56de\u7684\u5bf9\u8c61\u9700\u8981\u6709 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>bool IsCompleted {get;}<\/code><\/span><span class=\"md-plain\"> \u5c5e\u6027\uff0c\u548c <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>GetResult()<\/code><\/span><span class=\"md-plain\"> \u65b9\u6cd5\u3002<\/span><\/p>\n<\/li>\n<\/ul>\n<p class=\"md-end-block md-p md-focus\"><span class=\"md-plain\">\u77e5\u9053\u8fd9\u4e9b\u4ee5\u540e\uff0c\u6211\u4eec\u53ef\u4ee5\u5199\u51fa\u53ef\u7b49\u5f85\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>Lazy&lt;T&gt;<\/code><\/span><span class=\"md-plain\"> \u7c7b\uff1a<\/span><\/p>\n\n\n<pre class=\"wp-block-code\"><code>public struct LazyAwaiter&lt;T> : INotifyCompletion\r\n{\r\n    private readonly Lazy&lt;T> _lazy;\r\n \r\n    public LazyAwaiter(Lazy&lt;T> lazy) => _lazy = lazy;\r\n \r\n    public T GetResult() => _lazy.Value;\r\n \r\n    public bool IsCompleted => true;\r\n \r\n    public void OnCompleted(Action continuation) { }\r\n}\r\n \r\npublic static class LazyAwaiterExtensions\r\n{\r\n    public static LazyAwaiter&lt;T> GetAwaiter&lt;T>(this Lazy&lt;T> lazy)\r\n    {\r\n        return new LazyAwaiter&lt;T>(lazy);\r\n    }\r\n}<\/code><\/pre>\n\n\n\n<pre class=\"wp-block-code\"><code>public static async Task Foo()\r\n{\r\n    var lazy = new Lazy&lt;int>(() => 42);\r\n    var result = await lazy;\r\n    Console.WriteLine(result);\r\n}<\/code><\/pre>\n\n\n<p><span class=\"md-plain md-expand\">\u8fd9\u4e2a\u4f8b\u5b50\u770b\u8d77\u6765\u5341\u5206\u505a\u4f5c\uff0c\u4f46\u5b83\u6240\u5c55\u793a\u51fa\u7684\u53ef\u6269\u5c55\u6027\u7684\u8fd9\u4e00\u70b9\uff0c\u662f\u975e\u5e38\u6709\u7528\u4e14\u5728\u5f00\u53d1\u4e2d\u5e7f\u6cdb\u5e94\u7528\u7684\u3002\u6bd4\u5982\uff0c <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/github.com\/dotnet\/reactive\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">Reactive Extensions for .NET<\/span><\/a><\/span><span class=\"md-plain\"> \u9879\u76ee\u5c31\u63d0\u4f9b\u4e86 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/github.com\/dotnet\/reactive\/blob\/fa1629a1e12a8fc21c95aeff7863425c2485defd\/Rx.NET\/Source\/src\/System.Reactive\/Linq\/Observable.Awaiter.cs#L21\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">\u81ea\u5b9a\u4e49awaiter<\/span><\/a><\/span><span class=\"md-plain\"> \u6765\u5728\u5f02\u6b65\u65b9\u6cd5\u4e2d\u7b49\u5f85 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>IObservable&lt;T&gt;<\/code><\/span><span class=\"md-plain\"> \u5bf9\u8c61\u3002\u57fa\u7840\u7c7b\u5e93\uff08BCL\uff09\u81ea\u5df1\u4e5f\u63d0\u4f9b <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/referencesource.microsoft.com\/#mscorlib\/system\/runtime\/compilerservices\/YieldAwaitable.cs,45\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">YieldAwaitable<\/span><\/a><\/span><span class=\"md-plain\"> \u7ed9 <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/referencesource.microsoft.com\/#mscorlib\/system\/threading\/Tasks\/Task.cs,3031\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">Task.Yield<\/span><\/a><\/span><span class=\"md-plain\"> \u548c <\/span><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/referencesource.microsoft.com\/#mscorlib\/system\/security\/cryptography\/cryptostream.cs,328\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">HopToThreadPoolAwaitable<\/span><\/a><\/span><span class=\"md-plain md-expand\"> \u4f7f\u7528\uff1a<\/span><\/p>\n\n\n<pre class=\"wp-block-code\"><code>public struct HopToThreadPoolAwaitable : INotifyCompletion\r\n{\r\n    public HopToThreadPoolAwaitable GetAwaiter() => this;\r\n    public bool IsCompleted => false;\r\n \r\n    public void OnCompleted(Action continuation) => Task.Run(continuation);\r\n    public void GetResult() { }\r\n}<\/code><\/pre>\n\n\n<p><span class=\"md-plain md-expand\">\u6211\u4eec\u901a\u8fc7\u4e0b\u9762\u7684\u4f8b\u5b50\u6765\u6f14\u793a <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>HotToThreadPoolAwaitable<\/code><\/span><span class=\"md-plain md-expand\"> \u7684\u4f7f\u7528\uff1a<\/span><\/p>\n\n\n<pre class=\"wp-block-code\"><code>&#91;Test]\r\npublic async Task Test()\r\n{\r\n    var testThreadId = Thread.CurrentThread.ManagedThreadId;\r\n    await Sample();\r\n \r\n    async Task Sample()\r\n    {\r\n        Assert.AreEqual(Thread.CurrentThread.ManagedThreadId, testThreadId);\r\n \r\n        await default(HopToThreadPoolAwaitable);\r\n        Assert.AreNotEqual(Thread.CurrentThread.ManagedThreadId, testThreadId);\r\n    }\r\n}<\/code><\/pre>\n\n\n<p class=\"md-end-block md-p\"><span class=\"md-plain md-expand\">\u4efb\u4f55\u201c\u5f02\u6b65\u201d\u65b9\u6cd5\u7684\u7b2c\u4e00\u90e8\u5206\uff08\u7b2c\u4e00\u4e2a <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>await<\/code><\/span><span class=\"md-plain\"> \u5173\u952e\u5b57\u4e4b\u524d\u7684\u90e8\u5206\uff09\u90fd\u5df2\u540c\u6b65\u7684\u65b9\u5f0f\u6267\u884c\u3002\u5728\u5927\u591a\u6570\u60c5\u51b5\u4e0b\uff0c\u5bf9\u4e8e\u6025\u5207\u7684\u53c2\u6570\u6c42\u8bc1\u662f\u53ef\u53d6\u7684\uff0c\u4f46\u6709\u65f6\u5019\u6211\u4eec\u4e0d\u60f3\u8ba9\u88ab\u8c03\u7528\u65b9\u6cd5\u963b\u585e\u8c03\u7528\u8005\u7684\u7ebf\u7a0b\u3002<\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>HopToThreadPoolAwaitable<\/code><\/span><span class=\"md-plain\"> \u5c31\u80fd\u8ba9\u5f02\u6b65\u65b9\u6cd5\u5269\u4e0b\u7684\u90e8\u5206\u6d3e\u9001\u7ed9\u7ebf\u7a0b\u6c60\u4e2d\u7684\u5176\u4ed6\u7ebf\u7a0b\u6765\u6267\u884c\uff0c\u800c\u4e0d\u662f\u8c03\u7528\u8005\u7684\u7ebf\u7a0b\u3002<\/span><\/p>\n<h2 class=\"md-end-block md-heading\"><span class=\"md-plain\">Task-like \u7c7b\u578b<\/span><\/h2>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">C#\u7f16\u8bd1\u5668\u5f88\u65e9\u5c31\u652f\u6301\u4e86\u81ea\u5b9a\u4e49 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>awaiter<\/code><\/span><span class=\"md-plain\"> \uff08\u4eceC# 5\u5f00\u59cb\uff09\u3002\u8fd9\u9879\u6269\u5c55\u65b9\u5f0f\u786e\u5b9e\u6709\u7528\uff0c\u4f46\u4e5f\u6709\u9650\u5236\uff0c\u6240\u6709\u8fd4\u56de\u503c\u5fc5\u987b\u4e3a <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>void<\/code><\/span><span class=\"md-plain\"> \u3001<\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>Task<\/code><\/span><span class=\"md-plain\"> \u6216 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>Task&lt;T&gt;<\/code><\/span><span class=\"md-plain\"> \u3002\u4ece C# 7.2\u5f00\u59cb\uff0c\u7f16\u8bd1\u5668\u652f\u6301 task-like \u7c7b\u578b\u3002<\/span><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/github.com\/dotnet\/roslyn\/blob\/main\/docs\/features\/task-types.md\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">Task-like \u7c7b\u578b<\/span><\/a><\/span><span class=\"md-plain\"> \u662f\u7531 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncMethodBuilderAttribute<\/code><\/span><span class=\"md-plain\"> \u7279\u6027\u6807\u8bc6\u7684\u7c7b\u6216\u7ed3\u6784\u4f53\uff0c\u4e14\u8be5\u7279\u6027\u4e0e\u4e00\u4e2a <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>builder \u7c7b\u578b<\/code><\/span><span class=\"md-plain\"> \u5173\u8054\u3002 \u4e3a\u4e86\u8ba9 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain\"> \u7c7b\u578b\u53d1\u6325\u4f5c\u7528\uff0c\u5b83\u9700\u8981\u50cf\u4e0a\u4e00\u8282\u63d0\u5230\u7684\u4e00\u6837\u53ef\u88ab\u7b49\u5f85\u3002\u57fa\u672c\u4e0a\uff0c\u5728\u7b2c\u4e00\u8282\u7684\u65b9\u6cd5\u5f97\u5230\u5b98\u65b9\u652f\u6301\u540e\uff0c<\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain\"> \u7c7b\u578b\u5c31\u80fd\u591f\u8ba9\u524d\u9762\u4e24\u8282\u7684\u6269\u5c55\u70b9\u7ed3\u5408\u8d77\u6765\u3002<\/span><\/p>\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u6ce8\uff1a\u73b0\u5728\u4f60\u53ef\u4ee5\u81ea\u5b9a\u4e49\u4e0a\u6587\u63d0\u5230\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncMethodBuilderAttribute<\/code><\/span><span class=\"md-plain\"> \u3002\u4f60\u53ef\u4ee5\u5728\u6211\uff08\u4f5c\u8005\uff09\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>github \u4ed3\u5e93<\/code><\/span><span class=\"md-plain\"> \u627e\u5230\u4f8b\u5b50\u3002<\/span><\/p>\n<p class=\"md-end-block md-p md-focus\"><span class=\"md-plain\">\u4e0b\u9762\u6709\u4e00\u4e2a\u5c06 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain\"> \u5b9a\u4e49\u4e3a\u7ed3\u6784\u4f53\u7684\u7b80\u5355\u4f8b\u5b50\uff1a<\/span><\/p>\n\n\n<pre class=\"wp-block-code\"><code>public sealed class TaskLikeMethodBuilder\r\n{\r\n    public TaskLikeMethodBuilder()\r\n        => Console.WriteLine(\".ctor\");\r\n \r\n    public static TaskLikeMethodBuilder Create()\r\n        => new TaskLikeMethodBuilder();\r\n \r\n    public void SetResult() => Console.WriteLine(\"SetResult\");\r\n \r\n    public void Start&lt;TStateMachine>(ref TStateMachine stateMachine)\r\n        where TStateMachine : IAsyncStateMachine\r\n    {\r\n        Console.WriteLine(\"Start\");\r\n        stateMachine.MoveNext();\r\n    }\r\n \r\n    public TaskLike Task => default(TaskLike);\r\n \r\n    \/\/ AwaitOnCompleted, AwaitUnsafeOnCompleted, SetException \r\n    \/\/ and SetStateMachine are empty\r\n\r\n}\r\n \r\n&#91;System.Runtime.CompilerServices.AsyncMethodBuilder(typeof(TaskLikeMethodBuilder))]\r\npublic struct TaskLike\r\n{\r\n    public TaskLikeAwaiter GetAwaiter() => default(TaskLikeAwaiter);\r\n}\r\n \r\npublic struct TaskLikeAwaiter : INotifyCompletion\r\n{\r\n    public void GetResult() { }\r\n \r\n    public bool IsCompleted => true;\r\n \r\n    public void OnCompleted(Action continuation) { }\r\n}<\/code><\/pre>\n\n\n<p><span class=\"md-plain md-expand\">\u73b0\u5728\u6211\u4eec\u5c31\u80fd\u5b9a\u4e49\u4e00\u4e2a\u8fd4\u56de\u7c7b\u578b\u662f <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>TaskLike<\/code><\/span><span class=\"md-plain\"> \u7684\u65b9\u6cd5\uff0c\u751a\u81f3\u5728\u65b9\u6cd5\u5185\u6211\u4eec\u53ef\u4ee5\u4f7f\u7528\u4e0d\u540c\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain md-expand\"> \u7c7b\u578b\u3002<\/span><\/p>\n\n\n<pre class=\"wp-block-code\"><code>public async TaskLike FooAsync()\r\n{\r\n    await Task.Yield();\r\n    await default(TaskLike);\r\n}<\/code><\/pre>\n\n\n<p class=\"md-end-block md-p\"><span class=\"md-plain md-expand\">\u4f7f\u7528 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain\"> \u6700\u4e3b\u8981\u7684\u539f\u56e0\u662f\u5b83\u80fd\u591f\u51cf\u5c11\u4f7f\u7528\u5f02\u6b65\u65b9\u6cd5\u7684\u5f00\u9500\u3002\u6bcf\u4e00\u6b21\u4f7f\u7528\u8fd4\u56de <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>Task&lt;T&gt;<\/code><\/span><span class=\"md-plain\"> \u7684\u5f02\u6b65\u65b9\u6cd5\uff0c\u90fd\u81f3\u5c11\u4f1a\u5411\u5806\u7533\u8bf7\u4e00\u6b21\u5185\u5b58\u2014\u2014\u56e0\u4e3a <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task<\/code><\/span><span class=\"md-plain\"> \u5bf9\u8c61\u7684\u5b58\u5728\u3002\u8fd9\u5bf9\u4e8e\u5927\u90e8\u5206\u5e94\u7528\u7a0b\u5e8f\u6765\u8bb2\u8fd8\u5c06\u5c31\uff0c\u5c24\u5176\u662f\u7c97\u7c92\u5ea6\u5730\u4f7f\u7528\u5f02\u6b65\u65b9\u6cd5\u65f6\u3002\u4f46\u5bf9\u4e8e\u6bcf\u79d2\u53ef\u80fd\u8de8\u8d8a\u6570\u5343\u4e2a\u5c0f\u4efb\u52a1\u7684\u5e95\u5c42\u4ee3\u7801\u6765\u8bf4\uff0c\u90a3\u5c31\u6709\u4ebf\u70b9\u4e0d\u4f18\u96c5\u4e86\u3002\u5728\u8fd9\u4e9b\u573a\u666f\u4e0b\uff0c\u51cf\u5c11\u6bcf\u6b21\u8c03\u7528\u7684\u4e00\u6b21\u5806\u5185\u5b58\u7533\u8bf7\u53ef\u4ee5\u5408\u7406\u5730\u63d0\u9ad8\u6027\u80fd\u3002<\/span><\/p>\n<h2 class=\"md-end-block md-heading\"><span class=\"md-plain\">\u603b\u7ed3<\/span><\/h2>\n<ul class=\"ul-list\" data-mark=\"-\">\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">C# \u7f16\u8bd1\u5668\u63d0\u4f9b\u4e86\u591a\u79cd\u65b9\u6cd5\u6765\u8ba9\u6211\u4eec\u6269\u5c55\u5f02\u6b65\u65b9\u6cd5\u3002<\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u4f60\u53ef\u4ee5\u81ea\u5b9a\u4e49\u81ea\u5df1\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>AsyncTaskMethodBuilder<\/code><\/span><span class=\"md-plain\"> \u7c7b\u6765\u6539\u53d8 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>Task-base<\/code><\/span><span class=\"md-plain\"> \u5f02\u6b65\u65b9\u6cd5\u7684\u884c\u4e3a\u8868\u73b0\u3002<\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u4f60\u53ef\u4ee5\u901a\u8fc7\u5b9e\u73b0\u81ea\u5b9a\u4e49 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>awaiter<\/code><\/span><span class=\"md-plain\"> \u6765\u8ba9\u67d0\u4e2a\u5bf9\u8c61\u53ef\u88ab\u7b49\u5f85\u3002<\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-plain\">\u4ece C# 7 \u5f00\u59cb\uff0c\u4f60\u53ef\u4ee5\u81ea\u5b9a\u4e49\u81ea\u5df1\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain\"> \u7c7b\u3002<\/span><\/p>\n<\/li>\n<\/ul>\n<h2 class=\"md-end-block md-heading\"><span class=\"md-plain\">\u8bfe\u5916\u9605\u8bfb<\/span><\/h2>\n<ul class=\"ul-list\" data-mark=\"-\">\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-meta-i-c  md-link\"><a href=\"https:\/\/summid.icu\/index.php\/2021\/12\/11\/dissecting-the-async-methods-in-c\/\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">\u5256\u6790C#\u4e2d\u7684 async \u65b9\u6cd5<\/span><\/a><\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-meta-i-c md-link\"><a href=\"https:\/\/github.com\/SergeyTeplyakov\/EduAsync\/\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">\u5f02\u6b65\u65b9\u6cd5\u6559\u5b66\u9879\u76ee<\/span><\/a><\/span><\/p>\n<\/li>\n<li class=\"md-list-item\">\n<p class=\"md-end-block md-p\"><span class=\"md-meta-i-c md-link\"><a href=\"https:\/\/github.com\/dotnet\/roslyn\/blob\/main\/docs\/features\/task-types.md\" target=\"_blank\" rel=\"noopener\"><span class=\"md-plain\">Task-like \u7c7b\u578b<\/span><\/a><\/span><\/p>\n<\/li>\n<\/ul>\n<p class=\"md-end-block md-p md-focus\"><span class=\"md-plain\">\u4e0b\u6b21\uff0c\u6211\u4eec\u4f1a\u8ba8\u8bba\u5f71\u54cd\u5f02\u6b65\u65b9\u6cd5\u6027\u80fd\u7684\u77e5\u8bc6\u70b9\uff0c\u5e76\u4e14\u4ecb\u7ecd\u5b98\u65b9\u6700\u65b0\u7684 <\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>task-like<\/code><\/span><span class=\"md-plain\"> \u7c7b\u578b\u2014\u2014<\/span><span class=\"md-pair-s\" spellcheck=\"false\"><code>System.ValueTask<\/code><\/span><span class=\"md-plain\"> \u3002<\/span><\/p>","protected":false},"excerpt":{"rendered":"<p>\u672c\u6587\u624b\u52a8\u7ffb\u8bd1\u81ea\uff1aExtending the async methods in C# \u4e0a\u4e00\u7bc7\u6587\u7ae0\u4e2d\uff0c\u6211\u4eec\u8ba8\u8bba\u4e86  [&hellip;]<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[22],"tags":[23],"class_list":["post-1226","post","type-post","status-publish","format-standard","hentry","category-c","tag-c"],"_links":{"self":[{"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/posts\/1226","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/comments?post=1226"}],"version-history":[{"count":1,"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/posts\/1226\/revisions"}],"predecessor-version":[{"id":1227,"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/posts\/1226\/revisions\/1227"}],"wp:attachment":[{"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/media?parent=1226"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/categories?post=1226"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/summid.icu\/index.php\/wp-json\/wp\/v2\/tags?post=1226"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}