Loading [MathJax]/jax/input/TeX/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 删除。

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

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
[干货来袭]C#6.0新特性
本文介绍了C#6.0的一些新特性,包括自动属性初始化、方法函数支持lambda表达式、直接导入命名空间、扩展方法、非空语法糖、字符串格式化新玩法、索引初始化、异常过滤器以及异步等待的Catch块等。同时,对于某些不太容易理解的地方,作者也给出了相应的解释。
GuZhenYin
2018/01/04
8210
[干货来袭]C#6.0新特性
C#14新特性一览,你觉得实用吗?
在本文中,我们将一起探索即将随 C# 14(对应 .NET 10)发布的一些新特性。目前这些功能已可在 .NET 10 预览版(Preview 3)和 Visual Studio 17.14 或更高版本的预览版中进行体验。
郑子铭
2025/06/07
1010
C#14新特性一览,你觉得实用吗?
C#6.0 新增功能
C# 6.0 版本包含许多可提高开发人员工作效率的功能。 这些功能的总体效果是让你编写的代码更简洁、更具可读性。 该语法不像许多常见做法那样繁琐。 可以更轻松地看出设计意图。 好好了解这些功能可以帮助你提高生产力,编写更具可读性的代码。 你可以更专注于功能,而不是语言的构造。
张传宁IT讲堂
2019/09/17
2K0
C#6.0 新增功能
【深入浅出C#】章节 9: C#高级主题:反射和动态编程
反射和动态编程为开发人员提供了一组强大的工具,可以应对多样化的编程需求,提高代码的灵活性和可维护性。然而,应谨慎使用它们,以确保代码的安全性和性能。
喵叔
2023/09/03
1.3K0
[读书笔记]C#学习笔记五: C#3.0自动属性,匿名属性及扩展方法
前言 这一章算是看这本书最大的收获了, Lambda表达式让人用着屡试不爽, C#3.0可谓颠覆了我们的代码编写风格. 因为Lambda所需篇幅挺大, 所以先总结C#3.0智能编译器给我们带来的诸多好处, 下一遍会单独介绍Lambda表达式. 这篇主要包括的内容有: 自动属性,隐式类型,对象集合初始化,匿名类型,扩展方法. 下面一起来看下C#3.0 所带来的变化吧. 1,自动实现的属性 在C#3.0之前, 定义属性时一般会像下面这样去编写代码: 1 class Person 2 { 3 /
一枝花算不算浪漫
2018/05/18
8560
探索c#之Async、Await剖析
基本介绍 Async、Await是net4.x新增的异步编程方式,其目的是为了简化异步程序编写,和之前APM方式简单对比如下。 APM方式,BeginGetRequestStream需要传入回调函数,线程碰到BeginXXX时会以非阻塞形式继续执行下面逻辑,完成后回调先前传入的函数。 HttpWebRequest myReq =(HttpWebRequest)WebRequest.Create("http://cnblogs.com/"); myReq.BeginGetRequestSt
蘑菇先生
2018/05/21
8880
25个被低估的C#开发技巧:从性能优化到代码优雅的实战指南
我构建过从企业级应用到性能关键型系统的各种项目,然而,在这些年里,我注意到一件奇怪的事情——每个人都在谈论相同的最佳实践。
郑子铭
2025/05/01
2980
25个被低估的C#开发技巧:从性能优化到代码优雅的实战指南
.NET中那些所谓的新语法之一:自动属性、隐式类型、命名参数与自动初始化器
开篇:在日常的.NET开发学习中,我们往往会接触到一些较新的语法,它们相对以前的老语法相比,做了很多的改进,简化了很多繁杂的代码格式,也大大减少了我们这些菜鸟码农的代码量。但是,在开心欢乐之余,我们也不禁地对编译器内部到底为我们做了哪些事儿而感到好奇?于是,我们就借助反编译神器,去看看编译器到底做了啥事!其实本篇中很多都不算新语法,对于很多人来说可能都是接触了很久了,这里主要是针对.NET的老版本来说,是一个“相对”的新语法。
Edison Zhou
2018/08/20
8910
.NET中那些所谓的新语法之一:自动属性、隐式类型、命名参数与自动初始化器
C#历来语法特性总结
C# 11已与.NET 7一起发布,本文按照.NET的发布顺序,根据微软官方文档整理C#中一些有趣的语法特性。
郑子铭
2023/09/19
9301
C#历来语法特性总结
C# 6.0 功能预览 (二)
在Language Feature Status上面看到,其实更新的并不是特别多,为了不会误导看了C# 6.0 功能预览 (一)的园友,现在把官方的更新列表拿了过来,供大家参考 Roslyn 编译平台概述  示例 语法可视化工具 蓝图 语言功能状态 语言设计说明 FAQ 语言功能实现状态 存在: 在以前的版本中已经存在 完成: 已经在该版本中实现 计划: 预计在该版本中 可能: 预计可能在该版本中 撤销: 可能不在该版本中 没有: 该版本中没有 N/A: 没有意义 功能 例子 C#
数据分析
2018/03/01
7710
C#学习笔记——语法糖
语法糖(Syntactic sugar),也译为糖衣语法,是由英国计算机科学家彼得·约翰·兰达(Peter J. Landin)发明的一个术语,指计算机语言中添加的某种语法,这种语法对语言的功能并没有影响,但是更方便程序员使用。通常来说使用语法糖能够增加程序的可读性,从而减少程序代码出错的机会。需要声明的是“语法糖”这个词绝非贬义词,它可以给我们带来方便,是一种便捷的写法,编译器会帮我们做转换;而且可以提高开发编码的效率,在性能上也不会带来损失。
vv彭
2021/04/13
6800
掌握25个C#实战技巧:从代码优化到高效开发
C#持续进化,每个版本都引入新特性,助你编写更高效、更优雅的代码。掌握以下技巧,不仅能提升代码质量,还能让开发过程更愉悦。
郑子铭
2025/04/18
2520
掌握25个C#实战技巧:从代码优化到高效开发
C# 8.0的计划特性
虽然现在C# 7才发布不久,并且新的版本和特性还在增加中,但是C# 8.0已经为大家公开了一些未来可能出现的新特性。
句幽
2020/04/27
7250
C# 中参数验证方式的演变
一般在写方法的时候,第一步就是进行参数验证,这也体现了编码者的细心和缜密,但是在很多时候这个过程很枯燥和乏味,比如在拿到一个API设计文档的时候,通常会规定类型参数是否允许为空,如果是字符可能有长度限制,如果是整数可能需要判断范围,如果是一些特殊的类型比如电话号码,邮件地址等,可能需要使用正则表达式进行判断。
跟着阿笨一起玩NET
2018/09/19
1.7K0
C# 中参数验证方式的演变
C# 7.0简而言之 -- 01. C#和.NET Framework简介
C#里面所有的类型都有一个共享的基类, 这也意味之C#里面所有的类型都具备一些相同的基本功能, 例如任何类型都可以通过调用ToString()方法来转化成字符串.
solenovex
2018/04/26
1.7K2
C# 7.0简而言之 -- 01. C#和.NET Framework简介
C#语法之糖有点甜,语法之美很迷人
语法糖的出现绝对是广大码农的福音,为什么叫语法糖?简而言之就是你之前需要写10行代码,现在一行代码就可以搞定并且效率还不下降,你说甜不甜,鸡腿香不香?语法糖能够增加程序的可读性,从而减少程序代码出错的机会。下面介绍一下C#新增或者常用的语法糖。
苏州程序大白
2021/08/13
1.2K0
C#语法之糖有点甜,语法之美很迷人
一步一步学Linq to sql(一):预备知识
  Linq to sql(或者叫DLINQ)是LINQ(.NET语言集成查询)的一部分,全称基于关系数据的 .NET 语言集成查询,用于以对象形式管理关系数据,并提供了丰富的查询功能,它和Linq to xml、Linq to objects、Linq to dataset、Linq to entities等组成了强大的LINQ。
aehyok
2018/09/11
1.1K0
.NET中那些所谓的新语法之四:标准查询运算符与LINQ
开篇:在上一篇中,我们了解了预定义委托与Lambda表达式等所谓的新语法,这一篇我们继续征程,看看标准查询运算符和LINQ。标准查询运算符是定义在System.Linq.Enumerable类中的50多个为IEnumerable<T>准备的扩展方法,而LINQ则是一种类似于SQL风格的查询表达式,它们可以大大方便我们的日常开发工作。因此,需要我们予以关注起来!
Edison Zhou
2018/08/20
2.4K0
.NET中那些所谓的新语法之四:标准查询运算符与LINQ
C# 9.0新特性
CandidateFeaturesForCSharp9 看到标题,是不是认为我把标题写错了?是的,C# 8.0还未正式发布,在官网它的最新版本还是Preview 5,通往C#9的漫长道路却已经开始.前
码农阿宇
2019/06/14
1.7K0
C#编程 | 那些C#中很少人知道的科技
本文来告诉大家在C#很少有人会发现的科技。即使是工作了好多年的老司机也不一定会知道,如果觉得我在骗你,那么请看看下面。
Enjoy233
2021/12/23
7960
C#编程 | 那些C#中很少人知道的科技
相关推荐
[干货来袭]C#6.0新特性
更多 >
交个朋友
加入腾讯云技术交流站
洞悉AI新动向 Get大咖技术交流群
加入HAI高性能应用服务器交流群
探索HAI应用新境界 共享实践心得
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档