写点什么

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:114446
用户头像

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

关注

评论

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

看完这套Java笔记,才明白笔者同时斩获7份大厂offer是有原因的!

Java 程序员 后端

真是绝了!git标星19

Java 程序员 后端

男默女泪!全网疯传的华为内部网络协议神仙笔记究竟有何魅力?

Java 程序员 后端

疫情宅家4个月苦修,“不负年华“终斩阿里Java岗4面,分享下心路历程

Java 程序员 后端

真香!面试题库泄漏,在Github一夜爆火的面试题库,被各大厂要求直接下架

Java 程序员 后端

牛批!Java集合框架面试题精华集(2020最新版),附PDF版

Java 程序员 后端

白嫖!白嫖!【尚学堂】高淇Java300集全套学习资料

Java 程序员 后端

盘点阿里巴巴 34 个牛逼 GitHub 项目

Java 程序员 后端

看懂这篇文章就够了!什么是Redis缓存雪崩、缓存穿透和缓存击穿-五分钟统统搞定

Java 程序员 后端

看透了 Redis 源码之后,我总结出来的几点心得

Java 程序员 后端

爱了,在GitHub超火的Java程序性能优化实战笔记,实在太香!

Java 程序员 后端

独角兽余额宝(Java现场面试48题):性能调优+索引+Mysql+缓存

Java 程序员 后端

直播阿里远程面(钉钉部门),三面坐上“直通车

Java 程序员 后端

看看AQS阻塞队列和条件队列

Java 程序员 后端

知乎热问:国家何时整治程序员的高薪现象?太可怕了!

Java 程序员 后端

牛皮!华为工程师总结的Java生态知识体系面试必看笔记,太秀了

Java 程序员 后端

生产环境JVM内存溢出案例分析

Java 程序员 后端

用 Java 爬美女图片,这个厉害了!

Java 程序员 后端

玩转互联网公司面试!快手、爱奇艺、小红书等14家互联网企业面经分享!

Java 程序员 后端

盘点那些进行“网络编程”必须要知道的基础知识!

Java 程序员 后端

看阿里P8用5个核心技术和200个案例带你解密阿里亿级流量核心架构

Java 程序员 后端

电商中的SPU与SKU

Java 程序员 后端

看了这篇文章后,面试官再也不敢问你非结构化存储的原理了

Java 程序员 后端

看完这篇还不懂HashMap的扩容机制,那我要哭了~

Java 程序员 后端

牛!阿里巴巴首推的《Java进阶必备宝典》,从理论到实战

Java 程序员 后端

用好这几个技巧,解决Maven-Jar包冲突易如反掌

Java 程序员 后端

用时17天完成4面阿里,已斩获offer,总结下个人面经心得体会

Java 程序员 后端

看一遍就理解:动态规划详解

Java 程序员 后端

牛逼,这2招,彻底让你和 null say 拜拜

Java 程序员 后端

盘点郭德纲那些“催人奋进”的经典语录 -- 致被我们浪费的时光

Java 程序员 后端

直呼内行!阿里大佬离职带出内网专属“高并发系统设计”学习笔记

Java 程序员 后端

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