QCon 演讲火热征集中,快来分享技术实践与洞见! 了解详情
写点什么

Visual Basic 14 的语言特性

  • 2015-02-04
  • 本文字数:4633 字

    阅读完需:约 15 分钟

与 Visual Studio 类似,Visual Basic 也将从版本 12 直接跳到 14。虽然新版本中的许多特性对于 C#来说也是首次引进,但仍然有大量的功能增强是特别针对 VB 的,旨在简化 VB 的使用。本文列举了一些最令我们感兴趣的特性。

对 Null 的支持

新版本的一个特性是对 null 值的支持,该特性使用?. 操作符。这一特性与 C#相同,如果操作符左方的表达式返回值不为 null,则继续计算右方表达式。在处理外部资源所返回的少量数据时,该特性尤其有用。举例来说:

If customer.PrimaryResidence IsNot Nothing AndAlso customer.PrimaryResidence.Garage IsNot Nothing AndAlso property.PrimaryResidence.Garage.Cars = 2 Then Print(“Two Car Garage”)

这段代码将被简化成以下语句:

复制代码
If property.PrimaryResidence?.Garage?.Cars = 2 Then Print("Two Car Garage")

除此之外,还可以将该操作符与 If 操作符进行结合,实现为表达式提供某个默认值的功能:

复制代码
Dim numberOfCarPorts = If(property.PrimaryResidence?.Garage?.Cars, 0)

C#与 VB 并不是唯一两种支持这种 null 处理方式的语言。在 Apple 的产品中得到广泛应用的 Objective-C 语言默认就支持该行为。尤其是它的方法调用也使用. 操作符,其工作方式就类似于 VB 中的?. 操作符。

在 Objective-C 的社区中,人们对该特性的评价褒贬不一。某些开发者非常喜爱这项功能,因为他们在进行方法调用时无需担心空引用异常的产生。而另一些开发者则对此感到痛恨,因为在问题发生时,他们不会看到空引用异常的产生,只会看到方法调用失败。如此一来他们就会感到困惑,为什么方法返回了 null,而不是返回有效值或抛出异常。

元编程

在 Visual Basic 12 中,我们首次看到了 CallerNameAttribute 这一特性的引入。虽然这一特性解决了属性变更通知(property change notification)的问题,但它的通用性还不足以解决另外一部分问题,在这些问题中需要一个以字符串形式表达的唯一标识符,在这种情况下,需要使用到 NameOf 这个操作符。

以下这个示例是由来自 Visual Basic 团队的 Lucian Wischik 所提供的,其中包括对参数进行验证的逻辑。

复制代码
Function Lookup(name As String) As Customer
If name is Nothing Then Throw New ArgumentNullException(NameOf(name))

这种方式能够避免在修改了参数名称的时候,忘记修改了所抛出异常的构造函数中所定义的字符串。由于 NameOf 操作符实际上创建了一个常量,因此你可以在任何需要使用硬编码字符串的时候使用该操作符。

字符串插值(Interpolation)

自从十年以前.NET 初次问世的时候,String.Format 这个方法就要求开发者们对参数的数量进行计数。多年以来,由于计数错误所产生的 bug 可谓是不计其数。字符串插值这一技术最初是由 Mono 团队为 C#语言所创建的,它彻底解决了计数这种糟糕的做法。

插值字符串是由 $”开头的,而不是单单使用”。对于每个你需要插入值的位置,都要使用一对大括号进行转义,这一点与 String.Format 的做法是相同的。另一个与 String.Format 相同的地方在于可以在转义中加入格式化选项。在下面这个简单的示例中出现了两个变量,name 和 total,后者将被格式化为货币格式。

复制代码
Dim message = $"Hello {name}, your amount due is {total:C2}"

该语法本身就使用了 String.Format 方法,因此使用者同样需要注意在适当的场合进行转义,考虑一下以下字符串:

复制代码
Dim requestUrl = $"http://{server}/customer?name={customerName}"

这段代码会产生一个 bug,开发者实际上需要的是以下代码:

复制代码
Dim requestUrl = $"http://{server}/customer?name={UrlEncode(customerName)}"

FormattedString 对象

乍一看插值字符串的语法,似乎无法处理从外部资源中获取字符串的场景,例如从本地化表或资源字典中获取字符串。不过,微软正在努力实现这一功能。Lucian Wischik 写道:

不仅能够使用在不同的语言文化中,而且还能够从中抽取出原始的格式化字符串或者是参数(举例来说,如果你打算在 SQL 查询中使用该语法,或者会需要对参数进行转义,以避免产生字符串注入攻击)。但目前为止,我们还没有完全决定该语法的设计规格。

按照当前的规格声明草稿所说,插值字符串可以是一个常规的字符串,也可以由一个名为 FormattedString 的对象实现。当你试图将某个插值字符串赋值给一个实现了 IFormattable 接口的变量或是参数时,系统会自动创建一个 FormattedString 类型的实例。

该对象的 IFormattable.ToString 方法接受一个类型为 IFormatProvider 类型的参数,使用该参数能够重写格式化相关的行为。

复制代码
string IFormattable.ToString(string ignored, IFormatProvider formatProvider)
{
return String.Format(formatProvider, format, args);
}

在上面一段代码中,format 与 args 两个参数分别代表了待插值的原始字符串,以及它所对应的值。

多行字符串

在 VB 中新加入的一个特性是多行字符串。实现它不需要任何特殊的语法,只需要在希望分行的地方省略引号即可。根据源代码文件所使用的换行符的不同,该换行符会自动在 vbCrlf、vbCr 及 vbLf 等符号间进行选择。对于 Visual Studio 的用户来说,基本上都会选择 vbCrlf。

在目前,某些开发者会选择在 XML 文本中使用 CData 段落来模仿这一特性,这种方式虽然能够实现所需要的效果,但显得有些冗长与笨拙。

属性

自动属性现在可以标记为只读了。可以在声明时为该属性赋值,也可在构造函数中进行赋值。

该语法的使用方式应该不会出乎你的意料:

复制代码
Public ReadOnly FirstName As String = "Anonymous"
Public ReadOnly LastName As String
Public Sub New (firstName As String, lastName As String)
Me.FirstName = firstName
Me.LastName = lastName
End Sub

在使用这一特性时,应当考虑到某些特殊情况。要理解这些情况,你首先必须理解参数传递的 copy-in 和 copy-out 概念。CLR 只允许你为变量及字段进行引用传递(即 C#中的 ref 或 out 操作符)。但在 VB 中,你也能够为属性进行引用传递。

为了缓解这两者之间的分歧,VB 会在准备进行函数调用时创建一个本地变量,该属性的值会被拷贝到这个本地变量中。该本地变量随后被传递至函数中,函数体能够修改该本地变量的值。当该函数返回时,本地变量的值会拷贝回属性中。

在使用只读的自动属性时,将会应用以下规则:

  1. 如果你在构造函数中的某个 lambda 表达式中使用只读自动属性,编译器会提示语法错误。
  2. 如果在构造函数或初始化器中使用只读自动属性,将应用 copy-in 与 copy-out 规则。Copy-out 操作会将值写入系统为属性生成的字段中。
  3. 如果不在构造函数或初始化器中使用只读自动属性,则只会应用 copy-in 规则。Copy-out 操作根本不会发生,但也不会产生任何语法错误。

这些规则都是基于只读字段的工作原理所产生的。

注释

现在,在一个多行语句的每一行末尾都可以加入注释了。在之前的版本中,只能在多行语句的最后一行末尾加入注释。请看以下示例:

复制代码
Dim emailList =
From c in Customers
Where c.IsActive 'ignore inactive customers
And Not c.DoNotEmail 'we don’t need another spam violation
Select c.FullName, c.EmailAddress

结构体

结构体现在能够支持无参构造函数了。虽然 CLR 本身就支持这一特性,但还没有主流的编程语言实现了这一特性,其原因是构造函数的运行时机并不明确。举例来说,在创建某个结构体的数组时,该结构体的构造函数并不会运行。

如果你的代码是 myStruct = new MyStructure(),那很显然该构造函数会立即执行。而如果你的代码是 myStruct = Nothing,则显然不会执行构造函数。但在某个本地变量或成员变量自动初始化时又是否会执行构造函数呢?无论你选择哪一种答案,总会让一部分人感觉不爽。

数据文本(Data Literals)

从今年开始,数据文本(对于 JSON 格式来说非常重要的一个特性)终于改为使用符合 ISO 标准的格式了。在过去,数据文本一直使用基于美国的格式化形式,对于居住在欧洲的人来说就会产生一些迷惑。

  • 老风格:#3/4/2005#(是三月四日,还是四月三日?)
  • 新风格:#2005-4-3#

与 C#的互操作性

Overrides 修饰符将会隐含使用 Overloads 修饰符。在过去,VB 的开发者必须同时使用这两种修饰符,才能保证 C#的使用者在使用由 VB 所创建的类库时能够调用正确的重载方法。

接口模糊性

在 C#中使用接口继承这一特性时,会造成不易判断到底是哪个接口方法被调用的问题。在 VB 中不允许出现这种场景,但由于 C#允许这一特性,会造成出现某些 VB 无法实现的接口的情况。(在 Microsoft Dynamics 的某个产品中就数次出现这种情况。)

相对于 C#中所使用的“通过名称隐藏”(hide-by-name)的重载规则,VB 14 中将对这一限制进行放宽,转而使用一种(对 VB 来说)更传统的方式,即“通过签名隐藏”的规则。

命名空间解析

VB 也曾在命名空间解析这一问题上栽过跟斗,考虑一下以下代码:

Threading.Thread.Sleep(1000)

按 Lucian Wischik 所说:

之前,VB 会尝试查找“Threading”这一命名空间,由于它无法分辨 System.Threading 和 System.Windows.Threading 的区别,因此直接报错。现在,VB14 会同时支持这两种可能匹配的命名空间。如果你在代码编辑器输入 Threading.,那么在输入. 号之后,你会在智能提示中看到对这两个命名空间的支持。

类似的情况还有许多,举例来说:在编写 Winforms 应用时,ComponentModel.INotifyPropertyChanged 事件就会无法分辨 System.ComponentModel 及 System.Windows.Forms.ComponentModel,这一问题如今将不复存在。

TypeOf 和 IsNot

微软在十年前就创建了 IsNot 操作符,自那以来,就不断有 VB 的开发者要求微软允许在 TypeOf 表达式中使用 IsNot 操作符,举例如下:

复制代码
If TypeOf sender IsNot Button Then

预处理指令

VB 14 为预处理指令提供了两点改进之处。

Regi7on

Region 将能够在函数体中进行使用,甚至是跨两个函数体进行使用。

关闭警告

与 C#相同,Visual Basic 现在也能够关闭对某一个代码块的编译警告了。在规格说明中提供了一个示例:

复制代码
#Disable Warning BC42356 'suppress warning about no awaits in this method

通常来说,开发者会通过某个指令在该代码文件的其它地方重新打开这一警告

复制代码
#Enable Warning BC42356

如果该警告的 ID 中包含了空格或标点符号,则必须使用引号。微软的工具不会自动为你完成这一点,不过由 Roslyn 所编写的第三方分析器规则或许能实现这一点。

VB 的快速修复(Quick Fix)特性能够通过自动添加这些指令实现绕过某些警告的目的。这一点对于之前提到的第三方分析器规则来说尤其有用,因为你不一定能够很快地找到对应的 ID。

XML 文档验证

目前来说,VB 编译器会忽略 XML 文档的内容。而在 VB 14 中,编译器就会试图在文档中查找错误,例如不正确的参数引用名称。它还能够“正确地处理 crefs 标签中的泛型与操作符”。

部分模块(partial module)与接口声明

与类和结构体类型,你现在能够将模块与接口声明为部分(partial)了。通常来说,这一特性是为代码生成器所准备的,但也能够在跨多个平台分享代码时发挥作用。

关于作者

Jonathan Allen的第一份工作是在二十世纪 90 年代后期时,参与某个医疗诊所的管理信息系统项目的开发,将该项目由 Access 及 Excel 逐渐转化为企业级解决方案。随后,当他为某个商业部门的自动交易系统工作了五年之后,他决定转为进行高端用户界面的开发。在空余时间,他喜欢阅读及撰写一些关于西方武术在 15 世纪至 17 世纪之间发展的文章。

查看英文原文: Article: Visual Basic 14 Language Features

2015-02-04 23:114062
用户头像

发布了 428 篇内容, 共 180.7 次阅读, 收获喜欢 39 次。

关注

评论

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

人生算法:做好自己这家公司的CEO

石云升

读书笔记 思维模型 5月日更

智慧党建平台搭建,党建干部管理系统,智慧组工平台解决方案

英特尔院士斯旺:由外而内重塑芯片设计

E科讯

代码精进之路学习笔记

escray

学习 极客时间 5月日更

眼观六路耳听八方还不知疲倦?数仓智能运维服务体系是怎么做到的?

华为云开发者联盟

数据库 数据仓库 监控 智能运维 数据库监控

【Flutter 专题】118 图解特殊利器 ShaderMask 着色器

阿策小和尚

5月日更 Flutter 小菜 0 基础学习 Flutter Android 小菜鸟

低代码实现传统装饰企业的管理跃迁

华为云开发者联盟

低代码 华为云 计算 低代码开发 AppCube

从源码角度研究Java动态代理

叫我阿柒啊

动态代理 代理模式 rmi

五一假期旅游完突然收到(余额宝)面试,四面成功拿下offer

Java架构师迁哥

V8数据存储(上篇):栈和堆

梁龙先森

大前端 浏览器

为什么不推荐C++?

实力程序员

appium 入门参考

37手游iOS技术运营团队

ios 测试 自动化测试 iOS Developer

探秘区块链技术在计算机取证过程中的机制与应用

CECBC

一场“测谎”人机对战背后的故事:度小满的技术进击之路

脑极体

Django 之 Models(Models 模型 & 数据表关系)

若尘

django model Python编程 5月日更

并发王者课-青铜7:顺藤摸瓜-如何从synchronized中的锁认识Monitor

MetaThoughts

Java 多线程 并发

用图数据库可视化探索 Chia Network 区块链数据

古思为

区块链 可视化 图数据库

高可用DevHa实践,告诉你生产环境0性能故障是如何做到的!

TakinTalks稳定性社区

压测 性能调优 全链路压测 系统稳定高可用 性能压测

5 月 28 日 - 29 日阿里云峰会视频云专场直播预告

阿里云CloudImagine

阿里云 音视频

5分钟速读之Rust权威指南(十一)

wzx

rust

面向WEB开发人员的Docker(六):使用nginx部署静态网站

devpoint

Docker

密码学系列之:SAFER

程序那些事

密码学 程序那些事 SAFER

【玩转PDF】贼稳,产品要做一个三方合同签署,我方了!

牧小农

JVM

鸿蒙轻内核M核源码分析:数据结构之任务排序链表

华为云开发者联盟

鸿蒙 数据结构 任务排序链表 双向链表数组 鸿蒙轻内核

博睿数据2021战略发布巡展,开辟IT运维创新路径

博睿数据

博睿数据 数据链DNA 服务可达

week5作业

Geek_2e7dd7

架构实战营

计算社会科学 - DAY 17

Qien Z.

5月日更

webRTC的标准与发展

anyRTC开发者

音视频 WebRTC RTC

MySQL事务处理特性的实现原理

华为云开发者联盟

MySQL 数据库 innodb 事务 隔离

2021 全球技术领导力峰会 融云布道技术领导力进阶之路

融云 RongCloud

现在已经卷到需要问三色标记了吗?

艾小仙

Visual Basic 14的语言特性_语言 & 开发_Jonathan Allen_InfoQ精选文章