你所不知道的 C# 中的细节

简介:

你所不知道的 C# 中的细节
前言#
有一个东西叫做鸭子类型,所谓鸭子类型就是,只要一个东西表现得像鸭子那么就能推出这玩意就是鸭子。

C# 里面其实也暗藏了很多类似鸭子类型的东西,但是很多开发者并不知道,因此也就没法好好利用这些东西,那么今天我细数一下这些藏在编译器中的细节。

不是只有 Task 和 ValueTask 才能 await#
在 C# 中编写异步代码的时候,我们经常会选择将异步代码包含在一个 Task 或者 ValueTask 中,这样调用者就能用 await 的方式实现异步调用。

西卡西,并不是只有 Task 和 ValueTask 才能 await。Task 和 ValueTask 背后明明是由线程池参与调度的,可是为什么 C# 的 async/await 却被说成是 coroutine 呢?

因为你所 await 的东西不一定是 Task/ValueTask,在 C# 中只要你的类中包含 GetAwaiter() 方法和 bool IsCompleted 属性,并且 GetAwaiter() 返回的东西包含一个 GetResult() 方法、一个 bool IsCompleted 属性和实现了 INotifyCompletion,那么这个类的对象就是可以 await 的 。

因此在封装 I/O 操作的时候,我们可以自行实现一个 Awaiter,它基于底层的 epoll/IOCP 实现,这样当 await 的时候就不会创建出任何的线程,也不会出现任何的线程调度,而是直接让出控制权。而 OS 在完成 I/O 调用后通过 CompletionPort (Windows) 等通知用户态完成异步调用,此时恢复上下文继续执行剩余逻辑,这其实就是一个真正的 stackless coroutine。

Copy
public class MyTask
{

public MyAwaiter<T> GetAwaiter()
{
    return new MyAwaiter<T>();
}

}

public class MyAwaiter : INotifyCompletion
{

public bool IsCompleted { get; private set; }
public T GetResult()
{
    throw new NotImplementedException();
}
public void OnCompleted(Action continuation)
{
    throw new NotImplementedException();
}

}

public class Program
{

static async Task Main(string[] args)
{
    var obj = new MyTask<int>();
    await obj;
}

}
事实上,.NET Core 中的 I/O 相关的异步 API 也的确是这么做的,I/O 操作过程中是不会有任何线程分配等待结果的,都是 coroutine操作:I/O 操作开始后直接让出控制权,直到 I/O 操作完毕。而之所以有的时候你发现 await 前后线程变了,那只是因为 Task 本身被调度了。

UWP 开发中所用的 IAsyncAction/IAsyncOperation 则是来自底层的封装,和 Task 没有任何关系但是是可以 await 的,并且如果用 C++/WinRT 开发 UWP 的话,返回这些接口的方法也都是可以 co_await 的。

不是只有 IEnumerable 和 IEnumerator 才能被 foreach#
经常我们会写如下的代码:

Copy
foreach (var i in list)
{

// ......

}
然后一问为什么可以 foreach,大多都会回复因为这个 list 实现了 IEnumerable 或者 IEnumerator。

但是实际上,如果想要一个对象可被 foreach,只需要提供一个 GetEnumerator() 方法,并且 GetEnumerator() 返回的对象包含一个 bool MoveNext() 方法加一个 Current 属性即可。

Copy
class MyEnumerator
{

public T Current { get; private set; }
public bool MoveNext()
{
    throw new NotImplementedException();
}

}

class MyEnumerable
{

public MyEnumerator<T> GetEnumerator()
{
    throw new NotImplementedException();
}

}

class Program
{

public static void Main()
{
    var x = new MyEnumerable<int>();
    foreach (var i in x)
    {
        // ......
    }
}

}
不是只有 IAsyncEnumerable 和 IAsyncEnumerator 才能被 await foreach#
同上,但是这一次要求变了,GetEnumerator() 和 MoveNext() 变为 GetAsyncEnumerator() 和 MoveNextAsync()。

其中 MoveNextAsync() 返回的东西应该是一个 Awaitable,至于这个 Awaitable 到底是什么,它可以是 Task/ValueTask,也可以是其他的或者你自己实现的。

Copy
class MyAsyncEnumerator
{

public T Current { get; private set; }
public MyTask<bool> MoveNextAsync()
{
    throw new NotImplementedException();
}

}

class MyAsyncEnumerable
{

public MyAsyncEnumerator<T> GetAsyncEnumerator()
{
    throw new NotImplementedException();
}

}

class Program
{

public static async Task Main()
{
    var x = new MyAsyncEnumerable<int>();
    await foreach (var i in x)
    {
        // ......
    }
}

}
ref struct 要怎么实现 IDisposable#
众所周知 ref struct 因为必须在栈上且不能被装箱,所以不能实现接口,但是如果你的 ref struct 中有一个 void Dispose() 那么就可以用 using 语法实现对象的自动销毁。

Copy
ref struct MyDisposable
{

public void Dispose() => throw new NotImplementedException();

}

class Program
{

public static void Main()
{
    using var y = new MyDisposable();
    // ......
}

}
不是只有 Range 才能使用切片#
C# 8 引入了 Ranges,允许切片操作,但是其实并不是必须提供一个接收 Range 类型参数的 indexer 才能使用该特性。

只要你的类可以被计数(拥有 Length 或 Count 属性),并且可以被切片(拥有一个 Slice(int, int) 方法),那么就可以用该特性。

Copy
class MyRange
{

public int Count { get; private set; }
public object Slice(int x, int y) => throw new NotImplementedException();

}

class Program
{

public static void Main()
{
    var x = new MyRange();
    var y = x[1..];
}

}
不是只有 Index 才能使用索引#
C# 8 引入了 Indexes 用于索引,例如使用 ^1 索引倒数第一个元素,但是其实并不是必须提供一个接收 Index 类型参数的 indexer 才能使用该特性。

只要你的类可以被计数(拥有 Length 或 Count 属性),并且可以被索引(拥有一个接收 int 参数的索引器),那么就可以用该特性。

Copy
class MyIndex
{

public int Count { get; private set; }
public object this[int index]
{
    get => throw new NotImplementedException();
}

}

class Program
{

public static void Main()
{
    var x = new MyIndex();
    var y = x[^1];
}

}
作者: hez2010

出处:https://www.cnblogs.com/hez2010/p/12606419.html

相关文章
|
存储 缓存 监控
《优化接口设计的思路》系列:第二篇—接口用户上下文的设计与实现
大家好!我是sum墨,一个一线的底层码农,平时喜欢研究和思考一些技术相关的问题并整理成文,限于本人水平,如果文章和代码有表述不当之处,还请不吝赐教。 作为一名从业已达六年的老码农,我的工作主要是开发后端Java业务系统,包括各种管理后台和小程序等。在这些项目中,我设计过单/多租户体系系统,对接过许多开放平台,也搞过消息中心这类较为复杂的应用,但幸运的是,我至今还没有遇到过线上系统由于代码崩溃导致资损的情况。这其中的原因有三点:一是业务系统本身并不复杂;二是我一直遵循某大厂代码规约,在开发过程中尽可能按规约编写代码;三是经过多年的开发经验积累,我成为了一名熟练工,掌握了一些实用的技巧。
122 0
|
自然语言处理 Java 机器人
从细节出发:提高你的代码可读性
在编程的世界中,我们总是不断追求更高的性能,更优雅的设计,以及更复杂的特性。然而,我们不应忽视一个基本且重要的原则——代码的可读性。那么究竟何谓代码的可读性?顾名思义,代码可读性是指代码可理解的程度,是代码作者通过代码这个媒介,将需要表达的信息输出到读者脑中的能力。所以有的人说好的代码必然有清晰完整的注释,也有人说代码即注释,是代码简洁之道的最高境界,后者的观点飞哥持保留意见,毕竟真正能够做到代码即注释的有几人呢?
|
Java 编译器 应用服务中间件
代码开发优化细节
带有final修饰符的类是不可派生的。在Java核心API中,有许多应用final的例子,例如java.lang.String,整个类都是final的。为类指定final修饰符可以让类不可以被继承,为方法指定final修饰符可以让方法不可以被重写。如果指定了一个类为final,则该类所有的方法都是final的。Java编译器会寻找机会内联所有的final方法,内联对于提升Java运行效率作用重大,具体参见Java运行期优化。此举能够使性能平均提高50% 。
216 2
代码开发优化细节
|
C++
C++中需要注意的细节
C++中需要注意的细节
65 0
|
开发者 Python
盒图细节|学习笔记
快速学习盒图细节
113 0
盒图细节|学习笔记
|
JavaScript
使用组件的细节点
《Vue实战笔记》
90 1
|
测试技术 程序员
编程中你注意过这些细节吗?
阅读本文大概需要3分钟。
243 0
编程中你注意过这些细节吗?