Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >探索C#之6.0语法糖剖析

探索C#之6.0语法糖剖析

作者头像
蘑菇先生
发布于 2018-05-21 09:39:21
发布于 2018-05-21 09:39:21
1.7K00
代码可运行
举报
运行总次数:0
代码可运行

自动属性默认初始化

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public string Name { get; set; } = "hello world";

为了便于理解使用2.0语法展示,编译器生成代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 public class Customer 
{
 [CompilerGenerated] 
private string kBackingField = "hello world"; 
public Customer() 
{ 
this.kBackingField = "hello world"; 
}

public string Name
{
    [CompilerGenerated]
    get
    {
        return this.<Name>k__BackingField;
    }
    [CompilerGenerated]
    set
    {
        this.<Name>k__BackingField = value;
    }
}
} 

 从生成代码中可以看出编译器是在实例构造函数时,初始化属性信息的。

自动只读属性默认初始化

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public string Name1 { get; } = "hello world";

编译器生成代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
[CompilerGenerated] 
private readonly string kBackingField; 
public Customer() 
{
 this.kBackingField = "hello world";
 } 
public string Name1 
{
 [CompilerGenerated] 
get { return this.k__BackingField; }
 }

由于初始化默认值实在构造函数中赋值的,所以跟属性只读没关系。

表达式为主体的函数

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Body Get(int x, int y) => new Body(1 + x, 2 + y);

编译器生成如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
private Program.Body Get(int x, int y)
{
    return new Program.Body(1 + x, 2 + y);
}

简化了单行方法的编写,省去写大括号的功夫。

同时支持没有返回值的写法: 

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
void OutPut(int x, int y) => Console.WriteLine("hello world");

也支持异步函数的编写:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
async void OutPut(int x, int y) => await new Task(() => Console.WriteLine("hello wolrd"));

表达式为主体的属性(赋值)

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public string Name2 => "hello world";

编译器生成代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public string Name2 
{ 
get { return "mushroomsir"; }
 }

编译器只生成了个只读属性。

静态类导入

这个特性可以一次性导入某类型的所有静态成员,使静态成员在后面的代码中没有类型限制直接使用,像使用本类型下面的静态方法一样。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
using static System.Console;
 class Program 
{ 
static void Main(string[] args) 
{
 WriteLine("hello wolrd"); 
}
}

编译器生成代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
private static void Main(string[] args)
 {
 Console.WriteLine("hello wolrd"); 
}

省去了类型名称的重复编写。

Null条件运算符

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Customer customer = new Customer();
 string name3 = customer?.Name;

等同于:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Customer customer = new Customer();
if (customer1 != null)
{
    string name = customer1.Name;
}

可以和??组合起来使用:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
if (customer?.Face2()??false)

还可以2个一起用:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
int? Length = customer?.Name?.Length;

也可以方法调用:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
customer?.Face();

这个语法糖的目的是在对象使用前检查是否为null。如果对象为空,则赋值给变量为空值,所以例子中需要一个可以为空的int类型、即int?。

如果对象不为空,则调用对象的成员取值,并赋值给变量。

字符串格式化

String.Format有些不方便的地方是:必须输入"String.Format",使用{0}占位符、必须顺序来格式化、这点容易出错。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var s = String.Format("{0} is {1} year {{s}} old", p.Name, p.Age);

新的语法糖使用起来相对更轻松些:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var s = $"{p.Name} is {p.Age} year{{s}} old";

编译器生成如下,和之前没有区别:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var s = String.Format("{0} is {1} year{{s}} old", p.Name, p.Age);

有趣的是,新格式化方式还支持任何表达式的直接赋值:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var s = $"{p.Name} is {p.Age} year{(p.Age == 1 ? "" : "s")} old";

索引初始化

List虽然这样写可以编译通过,但是会抛异常的,使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var numbers = new List<string> { [7] = "seven", [9] = "nine", [13] = "thirteen" };

编译器生成代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
List list = new List(); 
list[7] = "seven";
 list[9] = "nine"; 
list[13] = "thirteen";

Dictionary可以执行,因为二者内部索引机制不一样:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 var numbers = new Dictionary<int, string> {[7] = "seven",[9] = "nine",[13] = "thirteen" };

编译器生成代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 Dictionary<int, string> dictionary2 = new Dictionary<int, string>();
    dictionary2[7] = "seven";
    dictionary2[9] = "nine";
    dictionary2[13] = "thirteen";
    Dictionary<int, string> dictionary = dictionary2;

异常过滤器when

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 try 
{ 
throw new ArgumentException("string error");
 }
 catch (ArgumentException e) when (myfilter(e))
 { 
Console.WriteLine(e.Message);
 }

static bool myfilter(ArgumentException e)
 { 
return false;
 }

When语法作用是:在进入到catch之前、验证when括号里myfilter方法返回的bool,如果返回true继续运行,false不走catch直接抛出异常。

使用这个filter可以更好的判断一个错误是继续处理还是重新抛出去。按照以前的做法,在catch块内如需再次抛出去,需要重新throw出去,这时的错误源是捕捉后在抛的,而不是原先的,有了when语法就可以直接定位到错误源。 

catch和finally代码块内的Await

Await异步处理是在c#5.0提出的,但不能在catch和finally代码块内使用,这次在C#6.0更新上支持了。

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    async void Solve()
    {
        try
        {
            await HttpMethodAsync();
        }
        catch (ArgumentException e)
        {
            await HttpMethodAsync();
        }
        finally
        {
            await HttpMethodAsync();
        }
    }

编译器把catch和finally的await生成到状态机里面的MoveNext()里面。原来里面只有 TaskAwaiter,现在多了2个。状态机里面的代码和原先的一样,只是更复杂了下,有兴趣的童鞋可以先看下Async、Await剖析再去深究。

nameof表达式

使用方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
string name = "";
Console.WriteLine(nameof(name));

控制台输出 "name"。

有时候会需要程序中一些成员的字符串名称,比如抛出ArgumentNullException异常的时候,想知道ArgumentNullException类型的字符串名称,这时候就可以用nameof获取字符

串“ArgumentNullException”。现在做法都是手动复制一下,但重构改名的时候容易忘记变更字符串,使用nameof就可以避免了。

当如下使用的时候,编译器会只取最后的ZipCode。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
nameof(person.Address.ZipCode)

编译器生成如下代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Console.WriteLine("name");

扩展方法

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    using static System.Linq.Enumerable; //引入类型,而不是命名空间
    class Program
    {
        static void Main()
        {
            var range = Range(5, 17);                // Ok: 不是扩展方法
            var odd = Where(range, i => i % 2 == 1); // Error, 不在全局作用域里
            var even = range.Where(i => i % 2 == 0); // Ok
        }
    }

首先Enumerable是个静态类,里面是各种扩展方法,比如range。static的作用是把类型的静态成员一次性导入,rang虽然是静态方法,但不能导入,比如where。

因为扩展方法虽然是一个静态方法,但是语法规定它作为一个实例方法使用(打点),所以不能在全局作用域里当静态方法用,因此var odd = Where(range, i => i % 2 == 1)是错误的。

但是static却能把类型的扩展方法作为扩展方法本身角色的功能导入进去,所以var even = range.Where(i => i % 2 == 0)是ok的。

这里可能稍微有点绕,lz尽量写清楚,static新用法有2个功能:

  1. 把静态成员导入,但扩展方法比较特殊、排除在外。这时static是c# 6.0的新功能。
  2. 等同于把扩展方法的命名空间导入,所以在集合上可以打点调用扩展方法。这是之前就有的功能,而不是把扩展方法转成单纯的静态方法导入使用。

总结

看到园子里有介绍的文章,一时来兴趣了,下班后安装个社区版就研究分享下。 虽然微软一直出新东西,但都是由下至上迭代的,所以学习起来是非常快的。

参考https://github.com/dotnet/roslyn/wiki/New-Language-Features-in-C%23-6#expression-bodied-function-members

探索C#之系列导航

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2015-07-22 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
​推荐一个神器!可以在浏览器运行 Node.js
在最新的 Google I/O 主题演讲中 stackblitz 向大家介绍了他们与 Next.js 和 Google 团队合作开发的在线 IDE WebContainers ,真的是非常惊艳。
小丑同学
2021/07/01
1.7K0
​推荐一个神器!可以在浏览器运行 Node.js
Bolt․new:AI 帮你自动写代码 自动运行并自动部署 而且全部在浏览器中完成
Bolt․new:AI 帮你自动写代码 自动运行并自动部署 而且全部在浏览器中完成
AI uTools
2024/10/29
5570
Bolt․new:AI 帮你自动写代码 自动运行并自动部署 而且全部在浏览器中完成
用网络构建网络?WebContainer是个啥?
今天来介绍一个 StackBlitz 与 Next.js 和 Google 团队合作的新技术。
薛定喵君
2021/06/10
3.4K0
Bolt.new 一句话建站,真的太疯狂了!开源 AI 全栈开发新体验!
兄弟们,给大家推荐一款最近爆火的开源编程工具!一款专供于网站开发的 Cursor,还包括环境配置运行,而且还有开源版本,它就是 Bolt.new。
Python兴趣圈
2024/10/10
2.4K0
Bolt.new 一句话建站,真的太疯狂了!开源 AI 全栈开发新体验!
神器推荐:在浏览器分析 npm 包
今天跟大家推荐一个神器 pkg-size.dev,可以直接在浏览器对 npm 包进行分析(包括占用大小、打包大小、间接依赖项等等)。它的目标是让像我们可以更轻松地探索 npm 生态系统。 对于包的使用者,pkg-size 可以帮助你:
ConardLi
2023/08/23
9390
神器推荐:在浏览器分析 npm 包
狼叔:聊聊 Node.js
前阵子我在知乎上回答了《2021前端会有什么新的变化?》,单篇33.8万的阅读量,还是不错的,说明大家非常关心前端的变化趋势,这里再与大家分享一下我对 Node.js 相关内容的看法。
五月君
2021/07/15
1.3K0
手摸手打造类码上掘金在线IDE(一)
最近几个月一直在跟在线IDE打交道,当然,高端一点咱也可以叫他低代码平台,毕竟这个词是流量密码,因为听着高端,看着大气,闻着。。。额,没味
用户7413032
2022/10/05
7570
手摸手打造类码上掘金在线IDE(一)
在浏览器中,把 Vite 跑起来了!
大家好,我是 ssh,前几天在推上冲浪的时候,看到 Francois Valdy 宣布他制作了 browser-vite,成功把 Vite 成功在浏览器中运行起来了。这引起了我的兴趣,如何把重度依赖 node 的一个 Vite 跑在浏览器上?接下来,就和我一起探索揭秘吧。
ssh_晨曦时梦见兮
2022/03/07
1.5K0
【保姆级】前端使用node.js基础教程
Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境,常用于服务器端编程。以下是一些 Node.js 中常用的命令:
肥晨
2024/03/22
3650
可以浏览器页面运行 node 代码里了!webContainer 发布 1.0
2021年5月12日, stackblitz 团队在 谷歌 I/O 上提到了一下的内容:
CandyTong
2023/02/18
9960
相比于 Node.js,Deno 和 Bun 到底能带来什么?
Deno 和 Bun 是两个想颠覆 Node.js 的新的 JavaScript 运行时。这两个替代方案凭借其对 JavaScript 服务器开发的创新设计在社区里也收获了不少的人气。但真正的问题是:相比于 Node.js,它们究竟能带来些什么呢?
ConardLi
2024/03/11
9310
相比于 Node.js,Deno 和 Bun 到底能带来什么?
可以浏览器页面运行 node 代码里了!webContainer 发布 1.0
2021年5月12日, stackblitz 团队在 谷歌 I/O 上提到了一下的内容:
CandyTong
2023/02/24
1.4K0
可以浏览器页面运行 node 代码里了!webContainer 发布 1.0
面向未来的前端开发模式 - 写于2021年
在之前,给大家介绍过webcontainer这个技术,就是可以让Node.js运行在浏览器中的技术
Peter谭金杰
2022/03/22
9320
面向未来的前端开发模式 - 写于2021年
Node.js的下一代浏览器和移动自动化测试框架-WebdriverIO
web自动化测试工具Selenium支持多种语言,如Python、Java、Ruby、JavaScript等,大多数测试人员使用最多的语言就是Python、Java了,并且很多技术社区与网上的学习资料查找起来很方便,很容易上手。但对于其他语言,比如JavaScript,相对来说资源就少很多了,针对JavaScript本篇将介绍一款在Node.js环境下的web与移动自动化测试框架 - WebdriverIO。
wangmcn
2024/07/18
3980
Node.js的下一代浏览器和移动自动化测试框架-WebdriverIO
Node.js 调试大法 : v8_inspector
Gcaufy
2017/05/18
2.2K0
Node.js 调试大法 : v8_inspector
下一代浏览器和移动自动化测试框架:WebdriverIO
今天给大家推荐一款基于Node.js编写且号称下一代浏览器和移动自动化测试框架:WebdriverIO
测试开发技术
2024/08/09
5200
下一代浏览器和移动自动化测试框架:WebdriverIO
Node.js安装与基本使用
结论:Node.js是一款软件,支持JS基本语法,可以提供web服务器,可以通过npm插件管理资源。
陶然同学
2023/02/27
1.2K0
Node.js安装与基本使用
方便快捷的调试 Node.js 程序
在调试程序时总是会遇到各种挑战。Node.js 的异步工作流为这一艰巨的过程增加了额外的复杂性。尽管 V8 引擎为了方便访问异步栈跟踪进行了一些更新,但是在很多数情况下,我们只会在程序主线程上遇到错误,这使得调试有些困难。同样,当我们的 Node.js 程序崩溃时,通常需要依靠一些复杂的 CLI 工具来分析核心转储[1]。
疯狂的技术宅
2020/08/19
1.8K0
Node.js命令介绍
Node.js是一个基于Chrome的V8引擎的JavaScript运行环境,它允许开发者在服务器端运行JavaScript代码。Node.js的强大之处在于其非阻塞I/O模型和事件驱动架构,使其轻量级且高效,尤其适用于数据密集型实时应用。在Node.js中,我们可以通过命令行工具来执行各种操作,下面我将介绍一些常用的Node.js命令。
china马斯克
2024/07/27
3040
初识Node.js
概述 相信对于专注javascript发展的同学来说,nodejs已经不是一个陌生的词眼。有关nodejs的相关资料网上已经铺天盖地。由于它的高并发特性,造就了其特殊的应用地位。 借用Node.js官网的定义:Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境。Node.js 使用了一个事件驱动、非阻塞式 I/O 的模型,使其轻量又高效。Node.js 的包管理器 npm。 相信之前有了解React Native的都知道,RN依赖Node.js环境,还有Angular.j
xiangzhihong
2018/02/05
2.4K0
初识Node.js
相关推荐
​推荐一个神器!可以在浏览器运行 Node.js
更多 >
交个朋友
加入腾讯云技术交流站
洞悉AI新动向 Get大咖技术交流群
加入HAI高性能应用服务器交流群
探索HAI应用新境界 共享实践心得
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验