元组 (ValueTuple)类型是值类型;元组元素是公共字段,可以使用任意数量的元素定义元组,
元组项命名准则
务必对使用元组语法声明的所有变量使用驼峰式大小写。
****考虑对所有元组项名称使用 Pascal 命名法。
为什么要使用元组?
有时,可能会发现,将数据元素组合在一起非常有用。例如,假设要处理国家/地区相关信息,如 2017 年世界上最贫穷的国家/地区马拉维。它的首都是利隆圭,人均国内生产总值 (GDP) 为 226.50 美元。显然,可以为此类数据声明一个类,但它并不是真正典型的名词/对象。它似乎更像是一组相关数据,而不是对象。当然,若要设置 Country 对象(举个例子),所含数据远不止 Name、Capital 和 GDP per capita 这些属性。也可以将每个数据元素存储在各个变量中,但这样做的结果是,数据元素相互没有任何关联。除了变量名称可能共用后缀或前缀之外,226.50 美元与马拉维将无任何关联。另一种方法是,将所有数据组合到一个字符串中。不过,这样做的缺点是,必须先分析各个数据元素,然后才能分别处理这些元素。最后一种方法是创建匿名类型,但这样做同样有局限性。其实,元组足以完全替代匿名类型.-----Mark Michaelis
一、元组声明和分配示例代码:
//元组写法1通过example2.Item来引用 var example1 = (1,2,3,4,5,"23",1,2,3,1,5,6,2,3); Console.WriteLine(example1.Item10); //元组写法2, 通过example3.变量名引用 var example2 = (exa1:1, exa2: 2,3,4,5,6); Console.WriteLine(example2.exa2); //元组写法3, 通过example3.变量名引用 左侧不允许弃元 (int age, string name) example3 = (3, "Dog3"); Console.WriteLine(example3.name); //元组写法4 相当于批量赋值 可以单独使用 变量 左侧不允许弃元 (string sr, bool sb, int sc) = ("4sr",true,1); Console.WriteLine(sr); //元组写法5 元组元素是公共字段 所以可以单独引用 var (exa51, exa52) = ("51f", 5.1); Console.WriteLine(exa51); //元组写法6 元组元素是公共字段 所以可以单独引用 var example6 = ("post office", 6.3); (string destination, double distance) = example6; Console.WriteLine(distance); //元组写法7 将元组分配到各个已预声明的变量中。 var exa71 = string.Empty; var exa72 = 0.0; var example7 = ("post office", 7.2); (exa71, exa72) = example7; Console.WriteLine(exa72); //元组写法8 将元组分配到各个已预声明的变量中。 string country; string capital; double gdpPerCapita; (country, capital, gdpPerCapita) =("Malawi", "Lilongwe", 226.50); System.Console.WriteLine( $"The poorest country in the world in 2017 was {country}, {capital}: {gdpPerCapita}"); //元组写法9 弃元 将未命名的元组分配到一个隐式类型化变量中, var countrInfo = ("Malawi", "Lilongwe", 226.50); (string name, _, double gdpPerCapit) = countrInfo; Console.WriteLine(gdpPerCapit);
部分内容来源:https://docs.microsoft.com/zh-cn/archive/msdn-magazine/2017/august/essential-net-csharp-7-0-tuples-explained
二、元组赋值和析构
C# 支持满足以下两个条件的元组类型之间的赋值:
1、两个元组类型有相同数量的元素
2、对于每个元组位置, 右侧元组元素的类型与左侧相应的元组元素的类型相同或可以隐式转换为左侧相应的元
组元素的类型
(int, double) t1 = (17, 3.14); (double First, double Second) t2 = (0.0, 1.0); t2 = t1; Console.WriteLine(t2);
三、比较运算符!= 和== 从 C# 7.3 开始支持
元组赋值和元组相等比较不会考虑字段名称。
同时满足以下两个条件时,两个元组可比较:
两个元组具有相同数量的元素。 例如,如果 t1 和 t2 具有不同数目的元素, t1 != t2 则不会进行编译。
对于每个元组位置,可以使用 == 和 != 运算符对左右侧元组操作数中的相应元素进行比较。 例如,
(1, (2, 3)) == ((1, 2), 3) 不会进行编译,因为 1 不可与 (1, 2) 比较。
(int a, byte b) left = (5, 10); (long a, int b) right = (5, 10); Console.WriteLine(left == right); // output: True Console.WriteLine(left != right); // output: False var t1 = (A: 5, B: 10); var t2 = (B: 5, A: 10); Console.WriteLine(t1 == t2); // output: True Console.WriteLine(t1 != t2); // output: False
== 和 != 运算符将以短路方式对元组进行比较。 也就是说,一旦遇见一对不相等的元素或达到元组的末尾,操
作将立即停止。 但是,在进行任何比较之前,将对所有元组元素进行计算,如以下示例所示:
Console.WriteLine((Display(1), Display(2)) == (Display(3), Display(4))); int Display(int s) { Console.WriteLine(s); return s; } // Output: // 1 // 2 // 3 // 4 // False
四、元组作为 out 参数
通常,你会将具有 out 参数的方法重构为返回元组的方法。 但是,在某些情况下, out 参数可以是元组类型。
下面的示例演示了如何将元组作为 out 参数使用:
var limitsLookup = new Dictionary<int, (int Min, int Max)>() { [2] = (4, 10), [4] = (10, 20), [6] = (0, 23) }; if (limitsLookup.TryGetValue(4, out (int Min, int Max) limits)) { Console.WriteLine($"Found limits: min is {limits.Min}, max is {limits.Max}"); } // Output: // Found limits: min is 10, max is 20
五、元组与 System.Tuple
System.ValueTuple 类型支持的 C# 元组不同于 System.Tuple 类型表示的元组。 主要区别如下:
ValueTuple 类型是值类型。 Tuple 类型是引用类型。
ValueTuple 类型是可变的。 Tuple 类型是不可变的。
ValueTuple 类型的数据成员是字段。 Tuple 类型的数据成员是属性。