当前位置:首页 > 科技  > 软件

.NET8 正式发布, C#12 新变化

来源: 责编: 时间:2023-11-17 17:14:44 219观看
导读在 .NET Conf 2023 大会上,.NET 8 正式发布了,.NET 8 是一个长期支持(LTS)版本,这意味着可以获得三年的支持和补丁。我们也计划将框架从 .NET Core3.1 升级到 8 ,关于如何升级等升级完成后再来分享。要使用 .NET 8 ,需要

ucp28资讯网——每日最新资讯28at.com

在  .NET Conf  2023  大会上,.NET 8 正式发布了,.NET 8 是一个长期支持(LTS)版本,这意味着可以获得三年的支持和补丁。我们也计划将框架从 .NET Core3.1 升级到 8 ,关于如何升级等升级完成后再来分享。ucp28资讯网——每日最新资讯28at.com

要使用 .NET 8 ,需要安装相关的 SDK,可以在这个地址进行下载:https://dotnet.microsoft.com/zh-cn/download/dotnet/8.0,或者将 VS2022 升级到 17.8 。ucp28资讯网——每日最新资讯28at.com

虽然 8 又带来了很多方面的增强,比如:人工智能、云原生、性能、native AOT  等,但我还是最关注  C# 语言和一些框架层面的变化,下面介绍下 C# 12 和框架中的我认为比较实用的新增功能,全部更新说明可以看官方文档:https://learn.microsoft.com/zh-cn/dotnet/core/whats-new/dotnet-8 。ucp28资讯网——每日最新资讯28at.com

序列化增强

其他类型的内置支持

  • 可以对附加类型:Half、Int128、UInt128 进行序列化,在 .NET 7 中对这些类型序列化时不会报错,但内容不能正常获取。
  • 可以对 ReadOnlyMemory、Memory类型进行序列化。
  • 当  T  的类型为  byte  时,序列化结果为  base64,否则为  json 数组。
using System.Text.Json;//输出:[65500,170141183460469231731687303715884105727,340282366920938463463374607431768211455]Console.WriteLine(JsonSerializer.Serialize(new object[] { Half.MaxValue, Int128.MaxValue, UInt128.MaxValue }));//输出:"AQIDBAUG"Console.WriteLine(JsonSerializer.Serialize<ReadOnlyMemory<byte>>(new byte[] { 1,2,3,4,5,6}));//输出:[1,2,3]Console.WriteLine(JsonSerializer.Serialize<Memory<int>>(new int[] { 1, 2, 3 }));

接口层次结构

IDerived value = new DerivedImplement { Base = 0, Derived = 1 };Console.WriteLine(JsonSerializer.Serialize(value));//输出:{"Base":0,"Derived":1}public interface IBase{    public int Base { get; set; }}public interface IDerived : IBase{    public int Derived { get; set; }}public class DerivedImplement : IDerived{    public int Base { get; set; }    public int Derived { get; set; }}

上面代码中 IDerived 接口继承了 IBase 接口后,就拥有两个属性了。ucp28资讯网——每日最新资讯28at.com

在之前的版本(3.1、6、7)中使用包含两个属性的接口 IDerived 来接收对象的实例化,然后进行序列化,得到的结果只有:{Derived":1} ,继承过来的属性 Base 不能被识别。ucp28资讯网——每日最新资讯28at.com

在 8 中得到了改进,可以得到期望的结果,值得注意的是,如果之前使用了变通方式来进行处理,升级后需要有针对性进行测试和调整。ucp28资讯网——每日最新资讯28at.com

命名策略

下图是 8  中序列化时对命名策略的支持:ucp28资讯网——每日最新资讯28at.com

ucp28资讯网——每日最新资讯28at.com

在之前的版本:3.1、6、7  中都只支持 CamelCase 。在  8  中新增的策略如下:ucp28资讯网——每日最新资讯28at.com

  • KebabCaseLower:小写中划线,例如:user-name。
  • KebabCaseUpper:大写中划线,例如:USER-NAME。
  • SnakeCaseLower:小写中划线,例如:user_name。
  • SnakeCaseUpper:大写中划线,例如:USER_NAME。
var options1 = new JsonSerializerOptions{    PropertyNamingPolicy = JsonNamingPolicy.KebabCaseLower,};var options2 = new JsonSerializerOptions{    PropertyNamingPolicy = JsonNamingPolicy.KebabCaseUpper,};var options3 = new JsonSerializerOptions{    PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower,};var options4 = new JsonSerializerOptions{    PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseUpper,};Console.WriteLine(JsonSerializer.Serialize(new UserInfo() { UserName = "oec2003" }, options1));Console.WriteLine(JsonSerializer.Serialize(new UserInfo() { UserName = "oec2003" }, options2));Console.WriteLine(JsonSerializer.Serialize(new UserInfo() { UserName = "oec2003" }, options3));Console.WriteLine(JsonSerializer.Serialize(new UserInfo() { UserName = "oec2003" }, options4));public class UserInfo{    public string? UserName { get; set; }}

结果如下:ucp28资讯网——每日最新资讯28at.com

ucp28资讯网——每日最新资讯28at.com

调用 API 直接获取到对象

现在有一个接口返回如下图中的数据:ucp28资讯网——每日最新资讯28at.com

ucp28资讯网——每日最新资讯28at.com

如果是在 8 以前的版本中获取该接口的数据,需要先获取到接口内容,然后进行反序列化,代码如下:ucp28资讯网——每日最新资讯28at.com

const string RequestUri = "http://localhost:5145/user";using var client = new HttpClient();var stream =await client.GetStreamAsync(RequestUri);//反序列化var users = JsonSerializer.DeserializeAsyncEnumerable<UserInfo>(stream);await foreach(UserInfo user in users){    Console.WriteLine($"姓名:{user.userName}");}Console.ReadKey();public record UserInfo(string userName);

在版本 8  中可以直接调用 GetFromJsonAsAsyncEnumerable 方法直接得到对象,无需进行反序列化:ucp28资讯网——每日最新资讯28at.com

const string RequestUri = "http://localhost:5145/user";using var client = new HttpClient();IAsyncEnumerable<UserInfo> users = client.GetFromJsonAsAsyncEnumerable<UserInfo>(RequestUri);await foreach (UserInfo user in users){    Console.WriteLine($"姓名: {user.userName}");}Console.ReadKey();public record UserInfo(string userName);

上面两种代码的结果一样,如下图:ucp28资讯网——每日最新资讯28at.com

ucp28资讯网——每日最新资讯28at.com

随机数增强

在 8 中对随机数类 Random 提供了 GetItems() 方法,可以根据指定的数量在提供的一个集合中随机抽取数据项生成一个新的集合:ucp28资讯网——每日最新资讯28at.com

ReadOnlySpan<string> colors = new[]{"Red","Green","Blue","Black"};string[] t1 = Random.Shared.GetItems(colors, 10);Console.WriteLine(JsonSerializer.Serialize(t1));//输出:["Black","Green","Blue","Blue","Green","Blue","Green","Black","Green","Blue"]//每次都会不一样Console.ReadKey();

通过 Random 提供的 Shuffle() 方法,可以将一个集合中的数据项的顺序打乱:ucp28资讯网——每日最新资讯28at.com

string[] colors = new[]{"Red","Green","Blue","Black"};Random.Shared.Shuffle(colors);Console.WriteLine(JsonSerializer.Serialize(colors));Console.ReadKey();

新增的提高性能的类型

新增了 FrozenDictionary<TKey,TValue> 和 FrozenSet,这两个类型在 System.Collections.Frozen 命名空间下,创建这两种类型的集合后,就不允许对键和值进行任何更改,因此可以实现更快的读取操作。ucp28资讯网——每日最新资讯28at.com

下面是使用 BenchmarkDotNet  对 FrozenDictionary 和 Dictionary 进行测试的代码:ucp28资讯网——每日最新资讯28at.com

BenchmarkRunner.Run<FrozenDicTest>();Console.ReadKey();[SimpleJob(RunStrategy.ColdStart, iterationCount:5)]public class FrozenDicTest{    public static Dictionary<string, string> dic = new() {        { "name1","oec2003"},        { "name2","oec2004"},        { "name3","oec2005"}    };    public static FrozenDictionary<string, string> fdic = dic.ToFrozenDictionary();    [Benchmark]    public void TestDic()    {        for (int i = 0; i < 100000000; i++)        {            dic.TryGetValue("name", out _);        }    }    [Benchmark]    public void TestFDic()    {        for (int i = 0; i < 100000000; i++)        {            fdic.TryGetValue("name", out _);        }    }}

从测试结果看,效果还是很明显的:ucp28资讯网——每日最新资讯28at.com

ucp28资讯网——每日最新资讯28at.com

新增的 System.Buffers.SearchValues类,可以用来进行字符串的查找和匹配,相比较  string  类型的操作,性能有大幅提升,下面还是用 BenchmarkDotNet 进行测试:ucp28资讯网——每日最新资讯28at.com

BenchmarkRunner.Run<SearchValuesTest>();Console.ReadKey();[SimpleJob(RunStrategy.ColdStart, iterationCount: 5)]public class SearchValuesTest{    [Benchmark]    public void TestString()    {        var str = "!@#$%^&*()_1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz";        for (int i = 0; i < 100000000; i++)        {            str.Contains("z");        }    }    [Benchmark]    public void TestSearchValues()    {        var sv = SearchValues.Create("!@#$%^&*()_1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"u8);        byte b = (byte)"z"[0];        for (int i = 0; i < 100000000; i++)        {            sv.Contains(b);        }    }}

从运行结果看,有大约 5 倍的的提升:ucp28资讯网——每日最新资讯28at.com

ucp28资讯网——每日最新资讯28at.com

依赖注入增强

在 8 之前的版本中,依赖注入写法如下:ucp28资讯网——每日最新资讯28at.com

var builder = WebApplication.CreateBuilder(args);builder.Services.AddTransient<IUser, UserA>(); var app = builder.Build();app.MapGet("/user", (IUser user) =>{    return $"hello , {user.GetName()}";});app.Run();internal interface IUser{    string GetName();}internal class UserA: IUser{    public string GetName() => "oec2003";}

如果 IUser 接口有两个实现,上面代码中的写法就只能获取到最后一个注册类的实例,要实现一个接口多个实现类的注入,还需要写一些额外的代码,比较繁琐。ucp28资讯网——每日最新资讯28at.com

版本 8 中添加了注入关键字,可以很方便实现,看下面代码:ucp28资讯网——每日最新资讯28at.com

var builder = WebApplication.CreateBuilder(args);builder.Services.AddKeyedSingleton<IUser, UserA>("A");builder.Services.AddKeyedSingleton<IUser, UserB>("B");var app = builder.Build();app.MapGet("/user1", ([FromKeyedServices("A")] IUser user) =>{    return $"hello , {user?.GetName()}";});app.MapGet("/user2", ([FromKeyedServices("B")] IUser user) =>{    return $"hello , {user?.GetName()}";});app.Run();internal interface IUser{    string GetName();}internal class UserA: IUser{    public string GetName() => "oec2003";}internal class UserB : IUser{    public string GetName() => "oec2004";}

本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-27991-0.html.NET8 正式发布, C#12 新变化

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 带你吃透Kafka的可靠性设计

下一篇: null 不好,我推荐你使用 Optional

标签:
  • 热门焦点
Top
Baidu
map