[译]C# 7系列,Part 1: Value Tuples 值元组
Mark Zhou写了很不错的一系列介绍C# 7的文章,虽然是2年多年前发布的,不过对于不熟悉C# 7特性的同学来说,仍然有很高的阅读价值。
原文:https://blogs.msdn.microsoft.com/mazhou/2017/05/26/c-7-series-part-1-value-tuples/
译文:
从今天开始,我将开始一个新的C# 7系列文章,介绍C# 7+的新语言特性。请注意,我说的不是C# 7.0,我说的是C# 7+,因为将会有一些小的版本(比如7.1、7.2)逐步引入新的特性(感谢Roslyn!),比如async Main和default literals。
Tuples
类System.Tuple提供了一种类型来表示类似属性包的键值对。当你想用一种数据结构来保存一个带有属性(元素)的对象,但又不想创建一个单独的类型的时候,你可以使用它。下面的代码展示了如何用它作为一个方法的返回值,这个返回值包含了学生姓名和年龄。
public Tuple<string, int> GetStudentInfo(string id)
{
// Search by ID and find the student.
return new Tuple<string, int>("Annie", );
}
可以看到,我返回了一个Tuple<string,int>的实例对象,它的第一个参数是name,第二个参数是age。之后,我们在代码中调用这个方法,像这样:
public void Test()
{
Tuple<string, int> info = GetStudentInfo("100-000-1000");
Console.WriteLine($"Name: {info.Item1}, Age: {info.Item2}");
}
你可以通过引用Item1和Item2来访问name和age。
Tuple类有一些明显的问题:
- 您需要使用ItemX这样的形式来访问属性,这样的属性名可能对调用者来说没有什么含义,如果我们可以使用类似info.Name和info.Age这样的形式来访问会比info.Item1和info.Item2更好。
- 最多只能有8个属性。如果需要更多,最后一个类型参数必须是另一个元组。这使得语法非常难以理解。
- Tuple是一种引用类型,不像其他基本类型(它们是大多数值类型),它分配在堆上,对于CPU密集型操作来说,它可能需要太多的对象创建/分配。
Value Tuples
C# 7.0引入了ValueTuple结构,它是Tuple对象的值类型表示。C#语言团队为这个值元组类型做了很多不错的事情,包括新的语法和许多特性(比如解构)。
下面是使用Value Tuples的重写版本,请注意,如果在你的项目中不能用ValueTuple类型,那你必须通过NuGet下载System.ValueTuple 4.3.0 NuGet包到你的项目。如果您使用的是.Net Framework 4.7或更高版本,或者.Net Standard Library 2.0或更高版本,你什么也不用做,ValueTuple已经包含在内了。
public (string, int) GetStudentInfo(string id)
{
// Search by ID and find the student.
return ("Annie", );
} public void Test()
{
(string, int) info = GetStudentInfo("100-000-1000");
Console.WriteLine($"Name: {info.Item1}, Age: {info.Item2}");
}
通过使用语法(),上面的代码得到了极大的简化。
您甚至可以为ValueTuple中的每个元素指定一个名称,如下所示:
public (string name, int age) GetStudentInfo(string id)
{
// Search by ID and find the student.
return (name: "Annie", age: );
} public void Test()
{
(string name, int age) info = GetStudentInfo("100-000-1000");
Console.WriteLine($"Name: {info.name}, Age: {info.age}");
}
帅!现在你的元组对象中的元素有了好的元数据,那么你就不需要来回检查确认返回/访问元素的顺序是正确的了。
当您使用值元组时,Visual Studio IDE会给您提示。
Value Tuple 解构
你可以从值元组对象中解构元素,并访问局部变量。
// 解构使用 var (x, y) 语法,
// 或者 (var x, var y) 语法。
var (name, age) = GetStudentInfo("100-000-1000");
// 现在你有两个局部变量:name and age.
Console.WriteLine($"Name: {name}, Age: {age}");
如果只关心某些元素而不是所有元素,可以使用_关键字忽略局部变量。
var (name, _) = GetStudentInfo("100-000-1000");
// 现在你只有一个局部变量:name,值age被忽略了。
Console.WriteLine($"Name: {name}");
从Value Tuples到Tuples
类型System.Tuple和System.ValueTuple提供了一些扩展方法来帮助它们之间相互转换。
var valueTuple = (id: , name: "Annie", age: , dob: DateTime.Parse("1/1/1993"));
var tuple = valueTuple.ToTuple();
结论
ValueTuple使C#语言更现代,更易于使用简化的语法。它解决了许多Tuple的问题:
- 值元组对象具有第一类语法支持,它简化了使用元组元素的代码。
- 您可以用一个名称与值元组的元素相关联,从而获得一定程度的设计阶段和编译器阶段的代码验证。
请注意,与元组元素相关联的名字不是一个运行时的元数据,即在运行时的实例值元组中不存在这样一个名称的属性/字段,属性的名称仍Item1、Item2等等,所有的元素名称仅存在设计和编译阶段。 - 你现在可以通过使用解构和_关键字灵活地访问所有元组元素,或者其中的一部分。
- 值元组类型是值类型,没有继承或其他特性,这意味着值元组具有更好的性能。
由于值元组元素的名称不是运行时的,所以在使用一些类库(如Newtonsoft)进行序列化时,必须小心使用元组类型。除非你更新过支持新元数据(TupleElementNameAttribute等)的类库,否则你会遇到bug。
系列文章:
- [译]C# 7系列,Part 1: Value Tuples 值元组 (本文)
- [译]C# 7系列,Part 2: Async Main 异步Main方法
- [译]C# 7系列,Part 3: Default Literals 默认文本表达式
- [译]C# 7系列,Part 4: Discards 弃元
- [译]C# 7系列,Part 5: private protected 访问修饰符
- [译]C# 7系列, Part 6: Read-only structs 只读结构
- [译]C# 7系列,Part 7: ref Returns ref返回结果
- [译]C# 7系列,Part 8: in Parameters in参数
- [译]C# 7系列,Part 9: ref structs ref结构
- [译]C# 7系列,Part 10: Span<T> and universal memory management Span<T>和统一内存管理 (本文)
[译]C# 7系列,Part 1: Value Tuples 值元组的更多相关文章
- [译]C# 7系列,Part 9: ref structs ref结构
原文:https://blogs.msdn.microsoft.com/mazhou/2018/03/02/c-7-series-part-9-ref-structs/ 背景 在之前的文章中,我解释了 ...
- [译]C# 7系列,Part 8: in Parameters in参数
原文:https://blogs.msdn.microsoft.com/mazhou/2018/01/08/c-7-series-part-8-in-parameters/ 背景 默认情况下,方法参数 ...
- [译]C# 7系列,Part 2: Async Main 异步Main方法
原文:https://blogs.msdn.microsoft.com/mazhou/2017/05/30/c-7-series-part-2-async-main/ 你大概知道,C#语言可以构建两种 ...
- [译]C# 7系列,Part 3: Default Literals 默认文本表达式
原文:https://blogs.msdn.microsoft.com/mazhou/2017/06/06/c-7-series-part-3-default-literals/ C#的default ...
- [译]C# 7系列,Part 4: Discards 弃元
原文:https://blogs.msdn.microsoft.com/mazhou/2017/06/27/c-7-series-part-4-discards/ 有时我们想要忽略一个方法返回的值,特 ...
- [译]C# 7系列,Part 5: private protected 访问修饰符
原文:https://blogs.msdn.microsoft.com/mazhou/2017/10/05/c-7-series-part-5-private-protected/ C#有几个可访问性 ...
- [译]C# 7系列,Part 6: Read-only structs 只读结构
原文:https://blogs.msdn.microsoft.com/mazhou/2017/11/21/c-7-series-part-6-read-only-structs/ 背景 在.NET世 ...
- [译]C# 7系列,Part 7: ref Returns ref返回结果
原文:https://blogs.msdn.microsoft.com/mazhou/2017/12/12/c-7-series-part-7-ref-returns/ 背景 有两种方法可以将一个值传 ...
- [译]C# 7系列,Part 10: Span<T> and universal memory management Span<T>和统一内存管理
原文:https://blogs.msdn.microsoft.com/mazhou/2018/03/25/c-7-series-part-10-spant-and-universal-memory- ...
随机推荐
- day 35 协程 IO多路复用
0.基于socket发送Http请求 import socket import requests # 方式一 ret = requests.get('https://www.baidu.com/s?w ...
- Java基础部分(11~20)
11."=="和 equals 方法究竟有什么区别? (单独把一个东西说清楚,然后再说清楚另一个,这样,它们的区别自然就出来了,混在一起说,则很难说清楚) ==操作符专门用来比较两 ...
- Clearcase Key commands check in code on linux
Supposed you are implemented done with all your codes(c is the alias for cleartool): New version add ...
- 全面解析JVM加载中初始化的时机
JVM类加载过程 JVM类加载过程分为几个阶段,分别是加载.验证.准备.解析和初始化.加载是把二进制字节码载入内存,验证是校验字节流中包含的信息是否符合当要求,准备是为静态变量分配内存并设置静态变量初 ...
- Ubuntu 18.04 LTS上安装NFS服务器和客户端
NFS是基于UDP/IP协议的应用,其实现主要是采用远程过程调用RPC机制,RPC提供了一组与机器.操作系统以及低层传送协议无关的存取远程文件的操作.RPC采用了XDR的支持.XDR是一种与机器无关的 ...
- Vue.js命名风格指南
前言 本命名风格指南推荐了一种统一的命名规范来编写 Vue.js 代码.这使得代码具有如下的特性: 统一团队的命名规范,其它开发者或是团队成员更容易上手阅读和理解. IDEs 更容易理解代码,从而提供 ...
- .NETCore 访问国产达梦数据库
前言 武汉达梦数据库有限公司成立于2000年,为中国电子信息产业集团(CEC)旗下基础软件企业,专业从事数据库管理系统的研发.销售与服务,同时可为用户提供大数据平台架构咨询.数据技术方案规划.产品部署 ...
- 制作通用framework的几点注意
一.创建framework,调成静态的framework . 二.匹配bitcode 三.增加-ObjC 在BuildSettting ->Linking->Other Linker Fl ...
- 声明式服务调用Feign
什么是 Feign Feign 是种声明式.模板化的 HTTP 客户端(仅在 consumer 中使用). 什么是声明式,有什么作用,解决什么问题? 声明式调用就像调用本地方法一样调用远程方法;无 ...
- numpy的基本API(一)——创建
numpy的基本创建API iwehdio的博客园:https://www.cnblogs.com/iwehdio/ 1.np.empty([a, b]) empty方法可以在无需初始化的情况下创建认 ...