[转]这些C#的运算符您都认识吗?


原文链接:【5min+】 这些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

>> 运算符将其左侧操作数向右移动:

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 条件运算符才会将成员访问 ?. 或元素访问 ?[] 运算应用于其操作数;否则,将返回 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循环然后再来截取原视数组的部分了。

类型测试运算符 is#

该关键字其实大家也很熟悉,但是在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 合并运算符 ?? 返回该值;否则,它会计算右操作数并返回其结果。如果左操作数的求值结果为非 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#版本,你可以使用所有的这些简写运算符,如果您使用的是以往的版本,请确保该运算符被支持哟。

本篇文章也不属于什么技术分享。不过有时候这些基础的东西往往会对咱们编码提供很大的便利性。

posted @   二次元攻城狮  阅读(343)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
  1. 1 烟花易冷 小柔Channel
  2. 2 红颜如霜 江壹纯
  3. 3 不谓侠 小桃Channel
  4. 4 小小恋歌 新坦结衣
  5. 5 神预言 袁娅维TIARAY
红颜如霜 - 江壹纯
00:00 / 00:00
An audio error has occurred, player will skip forward in 2 seconds.
点击右上角即可分享
微信分享提示
主题色彩