速来报名!AICon北京站鸿蒙专场~ 了解详情
写点什么

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

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

关注

评论

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

mac12系统怎么升级?苹果电脑macOS 12 Monterey系统离线安装包下载

Rose

零基础入门数据挖掘-课程汇总

阿里云天池

阿里云

科技进步对于我们的未来来说,到底是利好还是利弊?为什么?

算法的秘密

上云?!下云?!这难倒了孙悟空!

脑极体

云计算

【竞赛入门进阶】从赛题理解到竞赛入门基础

阿里云天池

阿里云

post为什么会发送两次请求?

百度搜索:蓝易云

云计算 Linux 运维 post 云服务器

Mac电池健康管理必备软件:AlDente Pro破解版 含AlDente Pro许可证

Rose

新零售SaaS架构:什么是线上商城系统?

快乐非自愿限量之名

架构 零售 SaaS

大模型 | LLM的7大主要功能有哪些?

澳鹏Appen

大模型 LLM

思维导图ai生成软件有哪些?这5款值得推荐!

彭宏豪95

人工智能 思维导图 在线白板 AIGC 思维导图软件

Photoshop 2024 安装激活教程 ps2024中文版 Mac/win

Rose

AppLink对51Tracking的集成方式

RestCloud

APPlink 自动化集成 51tracking

电子签赛道效率之争,e签宝率先给解法

ToB行业头条

如何使用Redis来防止穿透、击穿和雪崩问题

百度搜索:蓝易云

redis 云计算 Linux 运维 云服务器

Illustrator2021下载 AI 2021中文直装版 mac/win

Rose

XMind 2023思维导图软件 特别版/便携版

Rose

Prometheus多集群监控的3种方案,你选哪种?

华为云开发者联盟

后端 开发 华为云 华为云开发者联盟

设计原则 — DRY & Rule of three

Lemoon Can

设计原则 DRY Rule Of three

白嫖他悟空CRM项目 ,部署了直接用起来

程序猿忙什么

vue实现导出excel的多种方式

百度搜索:蓝易云

云计算 Linux 运维 Vue 云服务器

终端SSH工具SecureCRT mac激活版 含SecureCRT许可证

Rose

探索机器学习:从基础概念到应用实践

霍格沃兹测试开发学社

探索自然语言处理:语言模型的发展与应用

霍格沃兹测试开发学社

揭秘ChatGPT的Prompt方法:原理与应用总结

霍格沃兹测试开发学社

在 Amazon Bedrock 上使用 Anthropic Claude 系统 Prompt

亚马逊云科技 (Amazon Web Services)

Final Cut Pro X:Mac电脑上专业级剪辑软件 fcpx中文版下载

Rose

阿里通义灵码全面公测,来看看它的水平怎么样?

阿里巴巴云原生

阿里云 云原生 通义灵码

观测云在 .NET 业务中分析性能问题的最佳实践

观测云

APM Profile 可观测性

Java 异常处理与正则表达式详解,实例演练及最佳实践

小万哥

Java 程序人生 编程语言 软件工程 后端开发

云上三问,迈向智能时代的关键

脑极体

云计算

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