写点什么

C# 9 提案:模块初始化器

  • 2020-01-23
  • 本文字数:1239 字

    阅读完需:约 4 分钟

C# 9提案:模块初始化器

模块初始化器提案已经被提升到 C# 9 候选资格。它就像 C#中的静态构造函数,但不是应用于一个类,而是应用于整个程序集。


这个特性从一开始就存在于 CLR 中,但是直到现在还没有被 C#公开。根据模块初始化器提案,它将作为对静态构造函数语法的修改而公开。


[module: ModuleInitializer(typeof(MyModuleInitializer))]internal static class MyModuleInitializer{    static MyModuleInitializer()    {        // 将模块初始化器放在这里    }}
复制代码


从这个例子中可以看到,模块级属性会被标记为类名。然后,该类的静态构造函数被提升到模块初始化器级别。


这个特性可能会带来优于普通静态构造函数的性能。Mark Smeltzer 写道:


当前,运行时采用 init 锁,用于对是否已处理静态构造逻辑进行双重检查。即使向类中添加一个静态只读字段,也会立即增加该类中任何成员的每次外部使用的开销。


能够以一种有保证的、可预测的顺序运行初始化逻辑,并且在模块初始化后没有任何运行时操作,这将是一个巨大的优势。


另一个好处是模块初始化器是可预测的;其中的所有代码都是按顺序运行的。对于静态构造函数,从程序集的角度来看,它们的运行顺序是不确定的。根据客户端代码的不同,类 A 的构造函数可以在类 B 之前或之后运行。


对于上面的引文,Mark Smeltzer 在评论中进行了澄清:


模块初始化器仍然有上述的好处,但是在最初评论时,我并没有注意到.NET Core 3.0 对分层编译的一些最新改进…

.NET Core 3.0+中的分层编译解决了只读静态成员访问的问题。要了解更多信息,请查看 https://github.com/dotnet/coreclr/issues/24571#issuecomment-492401619。这个特性在.NET Core 3.0 版本已经发布。

这个特性非常棒:运行时最初会根据需要生成快速编译但不是最优的 JIT 代码,然后执行它。然后,在后台,运行时会分析上下文和 IL 代码,以确定是否可以实现更优的 JIT 解决方案。如果是,它将重新编译 IL 代码,并将低速的 JIT 路径替换为最优的 JIT 路径。当然,在实现的过程中会有一些额外的复杂性,但这是基本的思想。

对于静态类初始化器和静态只读字段初始化器,运行时将生成带有初始化锁的首遍(first pass)代码。那可以防止初始化器运行两次。该锁定还会带来运行时性能损失。因此,一旦初始化器运行一次,优化器就会生成新的经过优化的访问器代码路径,不再进行任何锁定!

再说一下,我不清楚实现细节(我可以确定,跟踪有关访问器肯定很复杂,需要在初始化之后进行优化),但净收益相当大:运行时本身自动优化掉了使用静态初始化器和静态只读字段的性能损失。


术语说明:


.NET CLR中的“模块”是一个包含 IL 代码的文件。“程序集”是由一个或多个模块组成的逻辑单元,其中一个模块被指定为头程序集。大多数.NET 语言被设计成只创建单模块/单文件程序集。因此,对大多数开发人员来说,这些术语是可以互换的。


VB中的“模块”就是 C#所称的“静态类”。


附属程序集”在某些方面类似于多模块/多文件程序集,但它是一个独立的概念。


原文链接


C# 9 Proposals: Module Initializers


2020-01-23 09:0012673

评论

发布
暂无评论
发现更多内容

建设 TiDB 自动化平台:转转 DBA 团队实践

PingCAP

数据库 自动化 TiDB

React源码解读之更新的创建

flyzz177

React

Unittest接口测试生成报告和日志方法

日志 单元测试 自动化测试 unittest 测试报告

Golang 编程“珠玑”

MatrixOrigin

golang 分布式数据库 编程语言 MatrixOrigin MatrixOne

美团前端必会vue面试题合集

bb_xiaxia1998

Vue

unittest中使用ddt后生成的测试报告名称如何修改?(如test_api_0修改成test_api_0_titile)

单元测试 自动化测试 unittest 参数化 ddt

万亿级对象存储的元数据系统架构设计和实践

百度开发者中心

对象存储 文件存储 百度沧海

假如问:你是怎样优化Vue项目的,该怎么回答

bb_xiaxia1998

Vue

前端手写面试题总结

helloworld1024fd

JavaScript

堡垒机行业标杆产品是哪家呢?有哪些功能?

行云管家

网络安全 信息安全 等保 堡垒机

React源码分析4-深度理解diff算法

goClient1992

React

构建云边端一体的分布式云架构,软硬结合驱动边缘计算创新场景

百度开发者中心

云原生 边缘计算 #百度智能云#

目前兰州市等保测评机构有几家?有新增的吗?

行云管家

等保 等级保护 等保测评 兰州

可视化分析能力MAX,瓴羊Quick BI带来全新数据分析体验

对不起该用户已成仙‖

谈谈Linux内核的噪声

统信软件

Linux 内核

腾讯前端一面经典手写面试题合集

helloworld1024fd

JavaScript

React源码分析3-render阶段(穿插scheduler和reconciler)

goClient1992

React

【FAQ】集成分析服务的常见问题及解决方案

HarmonyOS SDK

HMS Core

Led透明显示屏的发展超乎你想象

Dylan

LED 显示器 LED显示屏

LR性能测试常见问题及处理方法(一)

性能测试 问题排查 LoadRunner

解读Teradata结束中国直营背后的原因!国产数据库能填补空缺吗?

雨果

数据库管理工具 国产数据库 teradata SQL工具

react hook 源码完全解读

flyzz177

React

LR性能测试常见问题及处理方法(二)

性能测试 问题排查 LoadRunner

SQL工具性能实测:居然比Navicat还快,数百万行数据导出仅51秒

雨果

sql 数据库管理工具 Web SQL sql studio

写过vue自定义指令吗,原理是什么?.m

bb_xiaxia1998

Vue

React源码解读之任务调度

flyzz177

React

关于这个“微信提现”的问题,太炸裂了,以至于我写了段代码来验证!

why技术

Java 算法

自己手写一个redux

helloworld1024fd

JavaScript

React源码分析2-深入理解fiber

goClient1992

React

vivo x TiDB丨解决云服务海量数据挑战

PingCAP

TiDB

C# 9提案:模块初始化器_编程语言_Jonathan Allen_InfoQ精选文章