最近VS2019正式版发布了,装下来顺便试用了一下C#8.0,最大的看点应该就是可空引用类型了。不过C#8.0仍然处于Beta的状态,而且试用时也遇到了几个坑。

背景知识说明:

所谓的可空引用类型是指,一旦启用了可空引用类型这个新特征,引用类型将默认被视为不可空,无法赋予null,除非手工将它设为可空引用类型。

实战示例:

首先是新建一个C#的项目,在 项目文件(.csproj)里加入两行配置,目的是启用“C#8.0语言”和“可空引用类型”:

<LangVersion>8.0</LangVersion><NullableContextOptions>enable</NullableContextOptions>

整个文件看起来是这样的:

这样就算是整个项目全局启用了可空引用类型了。

注意

在VS2019正式版中,使用

<NullableContextOptions>enable</NullableContextOptions>

而不是使用

<NullableReferenceTypes>true</NullableReferenceTypes>

后者在正式版中已经失效了。

如果不希望全局启用可空引用类型的话,可以在程序代码中加入以下编译指令:

这样可以在加入了该指令的文件中,单独启用可空引用类型。但是,极度不推荐这种做法。为什么呢?因为该指令仅仅在该文件中有效,不能跨文件生效,从而无法阻止null逃逸到使用了该指令的文件中,也就是说,用了也等于没用。

一个很简单的例子足以证明:

注意上面项目文件中并没有全局启用可空引用类型,而下面的Class1.cs中使用了编译器指令来单独启用可空引用类型。

从运行结果可见,空引用仍然逃逸到使用了该指令的作用域中了。别说编译错误,连编译警告都没有。完全达不到理想的效果。

因此,强烈建议在项目文件中全局启用可空引用类型,而不是在某个源文件中单独使用。

另外,还有一点要注意的是,即使启用了可空引用类型后,默认情况下,即使对不可空引用赋予null,编译器也只会生成编译警告,而不是编译错误。仍然是能够编译通过的。一个大项目中,编译警告不可避免,甚至可能会很多,从而淹没了“给不可空引用类型赋予空值”这种不起眼的警告。

因此,建议将特定的警告视为错误。警告编号为8600、8625、8618、8604,或者将所有警告视为错误。具体是在项目文件中加入以下设置(见图一):

<WarningsAsErrors>8600 8625 8618 8604</WarningsAsErrors>

或者在项目编辑器中设置也可以:

这是我自己测试得出的结果,可能还有其它潜在的相关警告编号我没有测试出来。如果有谁知道的话,告诉我一下,谢谢。

做好这些配置之后,可以看到引用类型默认都不能赋予空值了:

这时候普通的引用类型的变量和参数都不能设为null了。

这样可以防止空值扩散开来,引起恼人的空引用异常。

但是,这里有个坑需要注意!!!!

struct里不适用可空引用的规则!!

struct里不适用可空引用的规则!!

struct里不适用可空引用的规则!!

这种情况下直接运行,仍然会抛出空引用异常!!!C#8.0仍未能完全封堵住空引用的逃逸。

其实我还是比较赞同用不可空引用类型的方案的,而不是可空引用类型的方案。毕竟我想要的,只不过是一个不可空的断言,只是想利用不可空引用来划分安全边界,从而防止空值的扩散。简单来说就是想将变量和参数明确声明为不可空引用类型。因为历史和现实的原因,大量的库都还没能全面使用可空引用类型。这种情况下,只有我使用可空引用类型,是不靠谱的。无法划分安全边界。

然而C#选择了可空引用类型的方案,而且还不是强制启用,而且默认只是警告。。跟没有一样。。。

比方说,我使用了一个第三方库项目,而空值的来源是正好是该库项目的,而我对这个库并没有源代码或者修改权限。这时候就无法阻止空值逃逸到我的项目中了。

还是之前的代码,只是稍微做一下修改。新增了一个库项目ClassLibrary1,这个库并没有使用可空引用类型。

库的代码如下:

很简单,就是LibClass3.GetInstance()本应该返回LibClass2的实例,但是出于某种原因,返回了null。但是我的项目中使用了LibClass2和LibClass3。我的项目是全局启用了可空引用类型的:

编译正常,毫无警告和错误。一旦运行,则抛出空引用异常:

可见,目前来说,C#8.0的可空引用类型并不能解决外源性的空值扩散,只能解决内源性的空值扩散,无法跨模块生效。还是洗洗睡吧。

参考资料:

https://docs.microsoft.com/en-us/dotnet/csharp/nullable-references

https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/nullable-types/index

https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/proposals/csharp-8.0/nullable-reference-types-specification#the-null-forgiving-operator

https://github.com/dotnet/roslyn/blob/master/docs/features/nullable-reference-types.md#setting-project-level-nullable-context

https://www.youtube.com/watch?v=VdC0aoa7ung

https://stackoverflow.com/questions/54852880/what-is-the-difference-between-nullablecontextoptions-and-nullablereferencetypes

原文地址:https://www.cnblogs.com/zlmdy/p/10656793.html

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

C#8.0可空引用类型的使用注意要点 1相关推荐

  1. C#8.0可空引用类型的使用注意要点

    C#8.0可空引用类型的使用注意要点 最近VS2019正式版发布了,装下来顺便试用了一下C#8.0,最大的看点应该就是可空引用类型了.不过C#8.0仍然处于Beta的状态,而且试用时也遇到了几个坑. ...

  2. 微软在C# 8中引入预览版可空引用类型

    微软已经为开发者提供了预览版的可空引用类型(Nullable Reference Type),想尝鲜的开发者可以尝试这个新特性,并提供反馈. \\ 预览版可空引用类型是Visual Studio 20 ...

  3. 如何将C# 7类库升级到C# 8?使用可空引用类型

    这篇文章将介绍将C# 7类库升级到C# 8(支持可空引用类型)的一个案例.本案例中使用的项目Tortuga Anchor由一组MVVM风格的基类.反射代码和各种实用程序函数组成.之所以选择这个项目,是 ...

  4. .NET 6新特性试用 | 可空引用类型

    前言 在查看<隐式using指令>功能时,我们在csproj中发现这样一个属性: 那么,Nullable到底是干嘛的? 可为空上下文 严格来说,这不是新特性,而是C# 8.0引入的特性之一 ...

  5. 为你的项目启用可空引用类型

    为你的项目启用可空引用类型 Intro C# 从 8.0 开始引入了可空引用类型,我们可以为项目启用可空引用类型来借助编译器来帮助我们更好的处理代码中的空引用的处理,可以避免我们写很多不必要 null ...

  6. 字符串分隔 -连续输入字符串,请按长度为8拆分每个字符串后输出到新的字符串数组; •长度不是8整数倍的字符串请在后面补数字0,空字符串不处理。...

    •连续输入字符串,请按长度为8拆分每个字符串后输出到新的字符串数组: •长度不是8整数倍的字符串请在后面补数字0,空字符串不处理. 输入描述: 连续输入字符串(输入2次,每个字符串长度小于100) 输 ...

  7. SqlServer 0和空字符串''等价?-----类型的隐式转换问题

    今天在修改一个bug的时候发现的一个比较奇怪的问题查询部门的时候偶尔会出现错误多查出一个不正确的部门 最后找到了原因 部门表 deptInfo 部门ID         部门名称 deptId     ...

  8. 关于是否在C#中加入不可空引用类型的争论

    来自微软的Mads Togersen在近期所提出的一条提议,即在C#语言中加入对不可空引用类型的支持在.NET社区中引起了热烈的争论.人们对此提议的反应大相径庭,既有人对此表示赞赏,也不乏倾向于保持现 ...

  9. C# 8 新特性 - 可空引用类型

    Nullable Reference Type. 在写C#代码的时候,你可能经常会遇到这个错误: 但如果想避免NullReferenceException的发生,确实需要做很多麻烦的工作. 可空引用类 ...

最新文章

  1. 一个bug隐藏了另外一个bug,reloaddata,
  2. Windows phone 7 Mango 更新发布会Session 整理
  3. PHP中全局变量$_POST[]和$_GET[]
  4. libxml2中处理中文
  5. 设置APP版本跟新提示
  6. rsync的配置文件模板及简单介绍,命令及参数
  7. python selenium 等待页面加载_python selenium 三种等待方式详解(实战常用)
  8. 记录重要的NLP学习资源链接
  9. 没经验能做软件测试吗?
  10. window 后台启动java参数启动
  11. 微软虚拟化技术——构建高效开发与测试环境
  12. java获取properties属性_java工具类中获取properties文件的属性
  13. asp.net937-图书馆座位管理系统
  14. Docker环境调优
  15. 材料成型计算机基础,材料成型及控制工程 主干课程
  16. 学校心理管理/预测系统
  17. oem是代工还是贴牌_OEM贴牌和ODM贴牌的区别
  18. Android代码安装apk程序
  19. 论文阅读:Channel Augmented Joint Learning for Visible-Infrared Recognition
  20. 前端的第一天(HTML)

热门文章

  1. 如何在Twitter上阻止令人讨厌的“今日热门新闻@yourname”垃圾邮件
  2. 计算机启动程序bios_如何构建自己的计算机,第三部分:准备BIOS
  3. 离开互联网行业_如何使用互联网再也不会离开家
  4. 怎么做mysql查询系统_mysql数据库系统学习(一)---一条SQL查询语句是如何执行的?...
  5. piwik抓取用户交互行为
  6. 安装debian的zabbix-agent客户端
  7. 2005-5-29+ 认识httphandler
  8. 如何使用 C# 判断一个文件是否为程序集
  9. 在ASP.Net Core和JAVA中,使用Azure配置密钥——Key Vault
  10. 如何分析 StackOverflow 异常 ?