【五分钟的dotnet】是一个利用您的碎片化时间来学习和丰富.net知识的博文系列。它所包含了.net体系中可能会涉及到的方方面面,好比C#的小细节,AspnetCore,微服务中的.net知识等等。
5min+不是超过5分钟的意思,"+"是知识的增长。so,它是让您花费5分钟如下的时间来提高您的知识储备量。程序员
此次更新好像隔的有点过久了。刚才在园子里看了一下我发的最后一篇文章,好吧,还停留在2020/01/24。原本计划春节停更一个星期的,结果Get到了一个有史以来最长的春节😂。c#
对于“ + ”,“ - ”,“ * ”,“ / ”,“ is ”,“ as ” 等C#的运算符可能你们都再熟悉不过了。可是有时候你们在阅读一些网站上的代码或者开源项目的源码时,会遇到一些奇奇怪怪的运算符,特别当几个少见的运算符连在一块儿的时候,仿佛有一种 “别人的C#和个人C#怎么不同” 的感受。数组
随着C#的版本更新,它为咱们提供了许许多多的语法糖和新运算符方便咱们更流畅的来编写代码。当有时候遇到不认识的运算符,就会本着 “百度不行就谷歌” 的程序员大法来疯狂操做一波。可是!! 搜索引擎他喵的竟然不认这些符号。微服务
因此,本文就整理了一些好玩儿的运算符作成了一个合集。若是碰到了不认识的操做符,也方便在这儿来查找。学习
先来看一段代码吧:测试
if (isFlagged) { _bits[propertyIndex / BitsPerInt] |= 1 << (propertyIndex % BitsPerInt); } else { _bits[propertyIndex / BitsPerInt] &= ~(1 << (propertyIndex % BitsPerInt)); } //节选自EF Core 中的结构体 “StateData”
有没有猛的一看感受已经不认识了的样子😭。毕竟对于我们平时开发应用层面的开发者来讲,不少位运算符不多用到,一下碰到了都忘记了什么意思。网站
~ 运算符经过反转每一个位产生其操做数的按位求补:搜索引擎
byte a = 10; // 二进制 0000 1010 var b = (byte)~a; // 二进制 1111 0101 。 b的十进制:245
<< 运算符将其左侧操做数向左移动:编码
byte a = 10; // 0000 1010 var b = (byte)a<<2; // 0010 1000。 b=40
>> 运算符将其左侧操做数向右移动.net
byte a = 10; // 0000 1010 var b = (byte)a<<2; // 0000 0010。 b=2
好比 10 * 2^3 当咱们用C#写的时候可能会写: 10 * Math.Pow(2,3) ,而如今能够直接写 10 << 3。
固然逻辑运算符还有其它的几个,好比 & 和 | ,这些平时你们用的比较多因此就很少写了。
^ 运算符计算其操做数的位逻辑异或:
byte a = 10; // 0000 1010 var b = a ^ 0b_0010_1011; // 0010 0001
因此若是配上我们C#的复合运算,好比 += , -=。相应的,上面的符号就能够写成 >>= ,|= , &= , ^= 等。
byte a = 10; a <<=2; // 40
没错,仍是这个符号。若是在索引器 [] 中使用它,它将充当索引的做用。
^ 运算符在 C# 8.0 和更高版本中提供,指示序列末尾的元素位置。例如,^1 指向序列的最后一个元素,^length 指向序列的第一个元素。
int[] xs = new[] { 0, 10, 20, 30, 40 }; int last = xs[^1]; Console.WriteLine(last); // output: 40
因此当咱们须要逆序来访问索引器的时候就不须要写成 : array[array.length - i] 了,直接^i 就能够了。
仅当操做数的计算结果为非 null 时,null 条件运算符才会将成员访问 ?. 或元素访问 ?[] 运算应用于其操做数;不然,将返回 null。
A?.B?.Do(C); A?.B?[C];
该操做符相信不少小伙伴早就使用起来了,毕竟能够直接省略掉咱们的 if(A==null),大幅提升了咱们的编码流畅度。
这个操做符很好玩,它是最新版本C#中才更新的。
.. 运算符在 C# 8.0 和更高版本中提供,指定索引范围的开头和末尾做为其操做数。
int[] numbers = new[] { 0, 10, 20, 30, 40, 50 }; int start = 1; int amountToTake = 3; int[] subset = numbers[start..(start + amountToTake)]; Display(subset); // output: 10 20 30
它能够结合上面的索引运算符 ^ 一同使用,好比:
int margin = 1; int[] inner = numbers[margin..^margin]; Display(inner); // output: 10 20 30 40
因此咱们就能够不用去for循环而后再来截取原视数组的部分了。
该关键字其实你们也很熟悉,可是在C# 7以后,它新增了:有模式匹配的类型测试。
object iBoxed = i; if (iBoxed is int a) { Console.WriteLine(a); }
在常规的类型测试表达式后面跟上一个变量,则该变量会转换为测试运算后的结果。这样就不须要咱们再去作一次判断了。
一说到这里,我忽然想到若是之后的C#可以支持这种表达式就行了,虽然只是YY😝:
if(cacl() != null) { var result = cacl(); //use result do something } //若是能更改成这样就行了 if(cacl() result !=null) { //use result do something }
该运算符也是很是有用的。若是左操做数的值不为 null,则 null 合并运算符 ?? 返回该值;不然,它会计算右操做数并返回其结果。 若是左操做数的求值结果为非 null,则 ?? 运算符不会对右操做数求值。
int? a = null; int b = a ?? -1; Console.WriteLine(b); // output: -1
这样就避免了咱们每次都会去写一个 if(xx = null),从而大大简化咱们的代码。
而且它还能够一直推算下去:
a ?? b ?? c
而在C# 8以后,??运算符还提供了合并运算的版本 ??= 。
b = b?? a; //等同于 b??=a;
因此我们常常对List判断是否为空,赋予初始值的操做,如今只须要一句话就完成了:
someList ??= new List<int>()).Add(5);
再结合上面的一些操做符来使用:
double SumNumbers(List<double[]> setsOfNumbers, int indexOfSetToSum) { return setsOfNumbers?[indexOfSetToSum]?.Sum() ?? double.NaN; }
因此如今我们再来看一些感受奇怪的符号就以为天然得多了,这些运算符不知道帮助我们省略掉了多少的if- else。
本文只是选取了一些不常见的运算符来介绍,而常见的 " | "、“ & ” ,“ ?. ” 等运算符相信你们用的也比较多了,因此就再也不说起。 还有就是关于指针的一些操做符,好比: " -> " , “ * ”等也没有涉及。
固然,若是您用的是最新的C#版本,你可使用全部的这些简写运算符,若是您使用的是以往的版本,请确保该运算符被支持哟。
本篇文章也不属于什么技术分享。不过有时候这些基础的东西每每会对我们编码提供很大的便利性。