最新文章 (全部类别)
.NETCore WebApi阻止接口重复调用(请求并发操作)
VS2022消除编译警告
“SymmetricAlgorithm.Create(string)”已过时:“Cryptographic factory methods accepting an algorithm name are obsolete. Use the parameterless Create factory method on the algorithm type instead
SHA256Managed/SHA512Managed已过时:Derived cryptographic types are obsolete. Use the Create method on the base type instead
MD5CryptoServiceProvider已过时:Derived cryptographic types are obsolete. Use the Create method on the base type instead
C#使用HttpClient获取IP地址位置和网络信息
判断IP是否是外网IP、内网IP
C#使用HttpClient获取公网IP
WebRequest.Create(string)已过时:WebRequest, HttpWebRequest, ServicePoint, and WebClient are obsolete. Use HttpClient instead
C#根据第三方提供的IP查询服务获取公网外网IP地址
html/dom/js/javascript开发记录
调试ASP.NETCore Web站点 - 清理IISExpress缓存数据(js,css)
EFCore+Oracle根据不同的Schema连接数据库
主程序集成CSFramework.EF 数据库框架(.NET7版本)
CSFramework.EF数据库框架简介(.NET8+EFCore)
迁移ECS服务器:导致ORACLE监听服务启动不了解决方案
SQLite数据库
VS2022编译报错:Visual Studio 容器工具需要 Docker Desktop
.NET 9 预览版+C#13新功能
EFCore禁用实体跟踪
WebApi开发框架V3.0 (.NETCore+EFCore) 增加AppSettings全局参数类
C#获取应用程序所有依赖的程序集
LINQ Expression 多条件复合条件组合(And/Or)
CSFrameworkV6客户案例 - MHR - 宁德时代制造人力资源系统
CS软件授权注册系统V3 - 发布证书
C/S软件授权注册系统V3.0(Winform+WebApi+.NET8+EFCore版本)
CS软件授权注册系统V3 - 购买方式
CS软件授权注册系统V3 - 试用版下载
CS软件授权注册系统-客户登记(制作证书)
C/S软件授权注册系统V3.0 - 管理员工具
CSFrameworkV6旗舰版开发框架 - 集成软件授权认证系统
CSFramework.Authentication 软件证书管理系统 - 制作软件客户授权证书
CSFramework.Authentication 软件证书管理系统 - MAC地址管理
CSFramework.Authentication 软件授权证书管理系统
Login/Logout接口调用dalUser的Login/Logout方法
C# Newtonsoft.Json.Linq.JObject 转对象
CSFramework.Authentication 软件授权认证系统 - 软件测试报告
C/S架构软件开发平台 - 旗舰版V6.0 - 底层框架迭代开发
C/S架构软件开发平台 - 旗舰版V6.1新功能 - 增加软件授权认证模块
C/S架构软件开发平台 - 旗舰版CSFrameworkV6 Bug修改记录
CS软件授权注册系统V3 - 开发手册 - 软件集成与用户注册
CS软件授权注册系统-模拟MES/ERP用户注册软件
CS软件授权注册系统-发布/部署WebApi服务器(IIS+.NET8+ASP.NETCore)
CS软件授权注册系统-VS2022调试WebApi接口
.NETCore Console控制台程序使用ILogger日志
CS软件授权注册系统-WebApi服务器介绍
ASP.NETCore集成Swagger添加Authorize按钮Bearer授权
CS软件授权注册系统-WebApi服务器配置
.NETCore WebApi发布到IIS服务器无法打开swagger
.NET8/ .NETCore /ASP.NETCore 部署WebApi到IIS服务器需要安装的运行环境
.net敏捷开发,创造卓越

C# OOP编程 模拟做早餐探索同步异步机制


C# OOP编程 模拟做早餐探索同步异步机制

概述

一天之计在于晨,每天的早餐也是必不可少,但是很多人为了节约时间,都是简单的吃点凑合一下或干脆不吃早餐,这对于个人身体和工作效率来说,无疑是不合理的,那么要如何做一顿早餐呢?如何能节约做早餐的时间呢?本文以一个简单的小例子,简述如何做一顿早餐及如何优化做早餐的时间。仅供学习分享使用,如有不足之处,还请指正。

正常情况下,做早餐可以分为以下几个步骤:

  1. 倒一杯咖啡。
  2. 加热平底锅,然后煎两个鸡蛋。
  3. 煎三片培根。
  4. 烤两片面包。
  5. 在烤面包上加黄油和果酱。
  6. 倒一杯橙汁。

同步方式做早餐

根据以上步骤进行编程,做一份早餐需要编写程序如下:

  1         /// <summary>
  2         /// 同步做早餐
  3         /// </summary>
  4         /// <param name="sender"></param>
  5         /// <param name="e"></param>
  6         private void btnBreakfast_Click(object sender, EventArgs e)
  7         {
  8             this.txtInfo.Clear();
  9             Stopwatch watch = Stopwatch.StartNew();
 10             watch.Start();
 11             //1. 倒一杯咖啡。
 12             string cup = PourCoffee();
 13             PrintInfo("咖啡冲好了");
 14             //2. 加热平底锅,然后煎两个鸡蛋。 
 15             string eggs = FryEggs(2);
 16             PrintInfo("鸡蛋煎好了");
 17             //3. 煎三片培根。 
 18             string bacon = FryBacon(3);
 19             PrintInfo("培根煎好了");
 20             //4. 烤两片面包。 
 21             string toast = ToastBread(2);
 22             //5. 在烤面包上加黄油和果酱。
 23             ApplyButter(toast);
 24             ApplyJam(toast);
 25             PrintInfo("面包烤好了");
 26             //6. 倒一杯橙汁。
 27             string oj = PourOJ();
 28             PrintInfo("橙汁倒好了");
 29             PrintInfo("早餐准备完毕!");
 30             watch.Stop();
 31             TimeSpan time = watch.Elapsed;
 32             PrintInfo(string.Format("总运行时间为:{0}秒", time.TotalSeconds.ToString("0.00")));
 33         }
 34 
 35         /// <summary>
 36         /// 倒一杯咖啡
 37         /// </summary>
 38         /// <returns></returns>
 39         private string PourCoffee()
 40         {
 41             PrintInfo("正在冲咖啡...");
 42             return "咖啡";
 43         }
 44 
 45         /// <summary>
 46         /// 抹果酱
 47         /// </summary>
 48         /// <param name="toast"></param>
 49         private void ApplyJam(string toast) =>
 50             PrintInfo("往面包抹果酱");
 51 
 52         /// <summary>
 53         /// 抹黄油
 54         /// </summary>
 55         /// <param name="toast"></param>
 56         private void ApplyButter(string toast) =>
 57             PrintInfo("往面包抹黄油");
 58 
 59         /// <summary>
 60         /// 烤面包
 61         /// </summary>
 62         /// <param name="slices"></param>
 63         /// <returns></returns>
 64         private string ToastBread(int slices)
 65         {
 66             for (int slice = 0; slice < slices; slice++)
 67             {
 68                 PrintInfo("往烤箱里面放面包");
 69             }
 70             PrintInfo("开始烤...");
 71             Task.Delay(3000).Wait();
 72             PrintInfo("从烤箱取出面包");
 73 
 74             return "烤面包";
 75         }
 76 
 77         /// <summary>
 78         /// 煎培根
 79         /// </summary>
 80         /// <param name="slices"></param>
 81         /// <returns></returns>
 82         private string FryBacon(int slices)
 83         {
 84             PrintInfo($"放 {slices} 片培根在平底锅");
 85             PrintInfo("煎第一片培根...");
 86             Task.Delay(3000).Wait();
 87             for (int slice = 0; slice < slices; slice++)
 88             {
 89                 PrintInfo("翻转培根");
 90             }
 91             PrintInfo("煎第二片培根...");
 92             Task.Delay(3000).Wait();
 93             PrintInfo("把培根放盘子里");
 94 
 95             return "煎培根";
 96         }
 97 
 98         /// <summary>
 99         /// 煎鸡蛋
100         /// </summary>
101         /// <param name="howMany"></param>
102         /// <returns></returns>
103         private string FryEggs(int howMany)
104         {
105             PrintInfo("加热平底锅...");
106             Task.Delay(3000).Wait();
107             PrintInfo($"磕开 {howMany} 个鸡蛋");
108             PrintInfo("煎鸡蛋 ...");
109             Task.Delay(3000).Wait();
110             PrintInfo("鸡蛋放盘子里");
111 
112             return "煎鸡蛋";
113         }
114 
115         /// <summary>
116         /// 倒橙汁
117         /// </summary>
118         /// <returns></returns>
119         private string PourOJ()
120         {
121             PrintInfo("倒一杯橙汁");
122             return "橙汁";
123         }

同步做早餐示例

通过运行示例,发现采用同步方式进行编程,做一份早餐,共计15秒钟,且在此15秒钟时间内,程序处于【卡住】状态,无法进行其他操作。如下所示:

C# 从做早餐看同步异步

同步做早餐示意图

同步方式做早餐,就是一个做完,再进行下一个,顺序执行,如下所示:

C# 从做早餐看同步异步

同步方式为何会【卡住】?

因为在程序进程中,会有一个主线程,用于响应用户的操作,同步方式下,做早餐的和前端页面同在主线程中,所以当开始做早餐时,就不能响应其他的操作了。这就是【两耳不闻窗外事,一心只读圣贤书】的境界。但如果让用户长时间处于等待状态,会让用户体验很不友好。比如,刘玄德三顾茅庐,大雪纷飞之下,诸葛亮在草庐中午睡,刘关张在大雪中静等。试问有几人会有玄德的耐心,何况程序也不是诸葛亮,用户也没有玄德的耐心!

异步方式做早餐

上述代码演示了不正确的实践:构造同步代码来执行异步操作。 顾名思义,此代码将阻止执行这段代码的线程执行任何其他操作。 在任何任务进行过程中,此代码也不会被中断。 就如同你将面包放进烤面包机后盯着此烤面包机一样。 你会无视任何跟你说话的人,直到面包弹出。如何做才能避免线程阻塞呢?答案就是异步。 await 关键字提供了一种非阻塞方式来启动任务,然后在此任务完成时继续执行。

首先更新代码,对于耗时的程序,采用异步方式做早餐,如下所示:

 1         private async void btnBreakfastAsync_Click(object sender, EventArgs e)
 2         {
 3             this.txtInfo.Clear();
 4             Stopwatch watch = Stopwatch.StartNew();
 5             watch.Start();
 6             //1. 倒一杯咖啡。
 7             string cup = PourCoffee();
 8             PrintInfo("咖啡冲好了");
 9             //2. 加热平底锅,然后煎两个鸡蛋。 
10             //Task<string> eggs = FryEggsAsync(2);
11             string eggs =await FryEggsAsync(2);
12             PrintInfo("鸡蛋煎好了");
13             //3. 煎三片培根。 
14             string bacon =await FryBaconAsync(3);
15             PrintInfo("培根煎好了");
16             //4. 烤两片面包。 
17             string toast =await ToastBreadAsync(2);
18             //5. 在烤面包上加黄油和果酱。
19             ApplyButter(toast);
20             ApplyJam(toast);
21             PrintInfo("面包烤好了");
22             //6. 倒一杯橙汁。
23             string oj = PourOJ();
24             PrintInfo("橙汁倒好了");
25             PrintInfo("早餐准备完毕!");
26             watch.Stop();
27             TimeSpan time = watch.Elapsed;
28             PrintInfo(string.Format("总运行时间为:{0}秒", time.TotalSeconds.ToString("0.00")));
29         }
30 
31         /// <summary>
32         /// 异步烤面包
33         /// </summary>
34         /// <param name="slices"></param>
35         /// <returns></returns>
36         private async Task<string> ToastBreadAsync(int slices)
37         {
38             for (int slice = 0; slice < slices; slice++)
39             {
40                 PrintInfo("往烤箱里面放面包");
41             }
42             PrintInfo("开始烤...");
43             await Task.Delay(3000);
44             PrintInfo("从烤箱取出面包");
45 
46             return "烤面包";
47         }
48 
49         /// <summary>
50         /// 异步煎培根
51         /// </summary>
52         /// <param name="slices"></param>
53         /// <returns></returns>
54         private async Task<string> FryBaconAsync(int slices)
55         {
56             PrintInfo($"放 {slices} 片培根在平底锅");
57             PrintInfo("煎第一片培根...");
58             await Task.Delay(3000);
59             for (int slice = 0; slice < slices; slice++)
60             {
61                 PrintInfo("翻转培根");
62             }
63             PrintInfo("煎第二片培根...");
64             await Task.Delay(3000);
65             PrintInfo("把培根放盘子里");
66 
67             return "煎培根";
68         }
69 
70         /// <summary>
71         /// 异步煎鸡蛋
72         /// </summary>
73         /// <param name="howMany"></param>
74         /// <returns></returns>
75         private async Task<string> FryEggsAsync(int howMany)
76         {
77             PrintInfo("加热平底锅...");
78             await Task.Delay(3000);
79             PrintInfo($"磕开 {howMany} 个鸡蛋");
80             PrintInfo("煎鸡蛋 ...");
81             await Task.Delay(3000);
82             PrintInfo("鸡蛋放盘子里");
83 
84             return "煎鸡蛋";
85         }

 注意:通过测试发现,异步方式和同步方式的执行时间一致,所以采用异步方式并不会缩短时间,但是程序已不再阻塞,可以同时响应用户的其他请求

优化异步做早餐

通过上述异步方式,虽然优化了程序,不再阻塞,但是时间并没有缩短,那么要如何优化程序来缩短时间,以便早早的吃上可口的早餐呢?答案就是在开始一个任务后,在等待任务完成时,可以继续进行准备其他的任务。 你也几乎将在同一时间完成所有工作。 你将吃到一顿热气腾腾的早餐。通过合并任务和调整任务的顺序,将大大节约任务的完成时间,如下所示:

 1         /// <summary>
 2         /// 优化异步做早餐
 3         /// </summary>
 4         /// <param name="sender"></param>
 5         /// <param name="e"></param>
 6         private async void btnBreakfast2_Click(object sender, EventArgs e)
 7         {
 8             this.txtInfo.Clear();
 9             Stopwatch watch = Stopwatch.StartNew();
10             watch.Start();
11             //1. 倒一杯咖啡。
12             string cup = PourCoffee();
13             PrintInfo("咖啡冲好了");
14             //2. 加热平底锅,然后煎两个鸡蛋。 
15             Task<string> eggsTask = FryEggsAsync(2);
16             //3. 煎三片培根。 
17             Task<string> baconTask = FryBaconAsync(3);
18             //4.5合起来 烤面包,抹果酱,黄油
19             Task<string> toastTask = MakeToastWithButterAndJamAsync(2);
20 
21             string eggs = await eggsTask;
22             PrintInfo("鸡蛋煎好了");
23 
24             string bacon = await baconTask;
25             PrintInfo("培根煎好了");
26 
27             string toast = await toastTask;
28             PrintInfo("面包烤好了");
29             //6. 倒一杯橙汁。
30             string oj = PourOJ();
31             PrintInfo("橙汁倒好了");
32             PrintInfo("早餐准备完毕!");
33             watch.Stop();
34             TimeSpan time = watch.Elapsed;
35             PrintInfo(string.Format("总运行时间为:{0}秒", time.TotalSeconds.ToString("0.00")));
36         }
37 
38         /// <summary>
39         /// 组合任务
40         /// </summary>
41         /// <param name="number"></param>
42         /// <returns></returns>
43         private async Task<string> MakeToastWithButterAndJamAsync(int number)
44         {
45             var toast = await ToastBreadAsync(number);
46             ApplyButter(toast);
47             ApplyJam(toast);
48             return toast;
49         }

在本例中,合并了【烤面包+抹果酱+抹黄油】为一个任务,这样是烤面包的同时,可以煎鸡蛋,煎培根,三项耗时任务同时执行。在三个任务都完成是,早餐也就做好了,示例如下所示:

C# 从做早餐看同步异步

通过以上优化示例发现,通过合并任务和调整顺序,做一份早餐,需要6.06秒。

优化异步早餐示意图

优化后的异步做早餐,由于一些任务并发运行,因此节约了时间。示意图如下所示:

C# 从做早餐看同步异步

异步异常

上述示例假定所有的任务都可以正常完成,那么如果某一个任务执行过程中发生了异常,要如何捕获呢?答案是:当任务无法成功完成时,它们将引发异常。 当启动的任务为 awaited 时,客户端代码可捕获这些异常。

例如当烤面包的时候,烤箱突然着火了,如何处理异常呢?代码如下所示:

 1         private async void btnBreakfastAsync3_Click(object sender, EventArgs e)
 2         {
 3             try
 4             {
 5                 this.txtInfo.Clear();
 6                 Stopwatch watch = Stopwatch.StartNew();
 7                 watch.Start();
 8                 //1. 倒一杯咖啡。
 9                 string cup = PourCoffee();
10                 PrintInfo("咖啡冲好了");
11                 //2. 加热平底锅,然后煎两个鸡蛋。 
12                 Task<string> eggsTask = FryEggsAsync(2);
13                 //3. 煎三片培根。 
14                 Task<string> baconTask = FryBaconAsync(3);
15                 //4.5合起来 烤面包,抹果酱,黄油
16                 Task<string> toastTask = MakeToastWithButterAndJamAsyncEx(2);
17 
18                 string eggs = await eggsTask;
19                 PrintInfo("鸡蛋煎好了");
20 
21                 string bacon = await baconTask;
22                 PrintInfo("培根煎好了");
23 
24                 string toast = await toastTask;
25                 PrintInfo("面包烤好了");
26                 //6. 倒一杯橙汁。
27                 string oj = PourOJ();
28                 PrintInfo("橙汁倒好了");
29                 PrintInfo("早餐准备完毕!");
30                 watch.Stop();
31                 TimeSpan time = watch.Elapsed;
32                 PrintInfo(string.Format("总运行时间为:{0}秒", time.TotalSeconds.ToString("0.00")));
33             }
34             catch (AggregateException ex) {
35                 PrintInfo("线程内部异常");
36                 PrintInfo(ex.StackTrace);
37             }
38             catch (Exception ex)
39             {
40                 PrintInfo("其他异常");
41                 PrintInfo(ex.Message);
42             }
43         }
44 
45         /// <summary>
46         /// 组合任务
47         /// </summary>
48         /// <param name="number"></param>
49         /// <returns></returns>
50         private async Task<string> MakeToastWithButterAndJamAsyncEx(int number)
51         {
52             var toast = await ToastBreadAsyncEx(number);
53             ApplyButter(toast);
54             ApplyJam(toast);
55             return toast;
56         }
57 
58         /// <summary>
59         /// 异步烤面包异常
60         /// </summary>
61         /// <param name="slices"></param>
62         /// <returns></returns>
63         private async Task<string> ToastBreadAsyncEx(int slices)
64         {
65             for (int slice = 0; slice < slices; slice++)
66             {
67                 PrintInfo("往烤箱里面放面包");
68             }
69             PrintInfo("开始烤...");
70             await Task.Delay(2000);
71             PrintInfo("着火了! 面包糊了!");
72             int a = 1, b = 0;
73             int i = a / b;//制造一个异常
74             //throw new InvalidOperationException("烤箱着火了!");
75             await Task.Delay(1000);
76             PrintInfo("从烤箱取出面包");
77 
78             return "烤面包";
79         }

异步任务异常示例

C# 从做早餐看同步异步

请注意,从烤面包机着火到发现异常,有相当多的任务要完成。 当异步运行的任务引发异常时,该任务出错。 Task 对象包含 Task.Exception 属性中引发的异常。 出错的任务在等待时引发异常。

需要理解两个重要机制:异常在出错的任务中的存储方式,以及在代码等待出错的任务时解包并重新引发异常的方式。

当异步运行的代码引发异常时,该异常存储在 Task 中。 Task.Exception 属性为 System.AggregateException,因为异步工作期间可能会引发多个异常。 引发的任何异常都将添加到 AggregateException.InnerExceptions 集合中。 如果该 Exception 属性为 NULL,则将创建一个新的 AggregateException 且引发的异常是该集合中的第一项。

对于出错的任务,最常见的情况是 Exception 属性只包含一个异常。 当代码 awaits 出错的任务时,将重新引发 AggregateException.InnerExceptions 集合中的第一个异常。 因此,此示例的输出显示 InvalidOperationException 而不是 AggregateException。 提取第一个内部异常使得使用异步方法与使用其对应的同步方法尽可能相似。 当你的场景可能生成多个异常时,可在代码中检查 Exception 属性。

高效的等待

通过以上示例,需要等待很多任务完成,然后早餐才算做好,那么如何才能高效优雅的等待呢?可以通过使用 Task 类的方法改进上述代码末尾的一系列 await 语句。其中一个 API 是 WhenAll,它将返回一个其参数列表中的所有任务都已完成时才完成的 Task,如下所示:

 1         private async void btnBreakfastAsync4_Click(object sender, EventArgs e)
 2         {
 3             this.txtInfo.Clear();
 4             Stopwatch watch = Stopwatch.StartNew();
 5             watch.Start();
 6             //1. 倒一杯咖啡。
 7             string cup = PourCoffee();
 8             PrintInfo("咖啡冲好了");
 9             //2. 加热平底锅,然后煎两个鸡蛋。 
10             Task<string> eggsTask = FryEggsAsync(2);
11             //3. 煎三片培根。 
12             Task<string> baconTask = FryBaconAsync(3);
13             //4.5合起来 烤面包,抹果酱,黄油
14             Task<string> toastTask = MakeToastWithButterAndJamAsync(2);
15             //等待任务完成
16             await Task.WhenAll(eggsTask, baconTask, toastTask);
17             
18             PrintInfo("鸡蛋煎好了");
19             PrintInfo("培根煎好了");
20             PrintInfo("面包烤好了");
21             //6. 倒一杯橙汁。
22             string oj = PourOJ();
23             PrintInfo("橙汁倒好了");
24             PrintInfo("早餐准备完毕!");
25             watch.Stop();
26             TimeSpan time = watch.Elapsed;
27             PrintInfo(string.Format("总运行时间为:{0}秒", time.TotalSeconds.ToString("0.00")));
28         }

另一种选择是使用 WhenAny,它将返回一个当其参数完成时才完成的 Task<Task>。如下所示:

 1         private async void btnBreakfastAsync5_Click(object sender, EventArgs e)
 2         {
 3             this.txtInfo.Clear();
 4             Stopwatch watch = Stopwatch.StartNew();
 5             watch.Start();
 6             //1. 倒一杯咖啡。
 7             string cup = PourCoffee();
 8             PrintInfo("咖啡冲好了");
 9             //2. 加热平底锅,然后煎两个鸡蛋。 
10             Task<string> eggsTask = FryEggsAsync(2);
11             //3. 煎三片培根。 
12             Task<string> baconTask = FryBaconAsync(3);
13             //4.5合起来 烤面包,抹果酱,黄油
14             Task<string> toastTask = MakeToastWithButterAndJamAsync(2);
15             //等待任务完成
16             var breakfastTasks = new List<Task> { eggsTask, baconTask, toastTask };
17             while (breakfastTasks.Count > 0)
18             {
19                 Task finishedTask = await Task.WhenAny(breakfastTasks);
20                 if (finishedTask == eggsTask)
21                 {
22                     PrintInfo("鸡蛋煎好了");
23                 }
24                 else if (finishedTask == baconTask)
25                 {
26                     PrintInfo("培根煎好了");
27                 }
28                 else if (finishedTask == toastTask)
29                 {
30                     PrintInfo("面包烤好了");
31                 }
32                 breakfastTasks.Remove(finishedTask);
33             }
34             //6. 倒一杯橙汁。
35             string oj = PourOJ();
36             PrintInfo("橙汁倒好了");
37             PrintInfo("早餐准备完毕!");
38             watch.Stop();
39             TimeSpan time = watch.Elapsed;
40             PrintInfo(string.Format("总运行时间为:{0}秒", time.TotalSeconds.ToString("0.00")));
41         }

以上就是由同步到异步再到优化异步任务的逐步过程,旨在抛砖引玉,一起学习,共同进步。

 

版权声明:本文为开发框架文库发布内容,转载请附上原文出处连接
C/S框架网
上一篇:程序员在面试时,如何回答未来规划的问题
下一篇:ASP.NET MVC快速入门(一)
评论列表

发表评论

评论内容
昵称:
关联文章

C# OOP编程 模拟早餐探索同步异步机制
C#异步编程(多线程)
OOP:面向对象编程
OOP面向对象编程(1)图说OOP基础
C#面向对象设计(OOP)之对象多态(Polymorphic)
C#.NET 消息机制
C#串口通信编程
C# WebService异步处理/异步调用详解
使用OOP用什么好处:
C#.NET CLR垃圾回收机制
异步实现Socket Server&Client
最全C#OOP设计思想之汽车模型源代码
C# 对象多态性(OOP基础)
什么是OOP
C#.Net OOP系列之接口设计及策略应用实战
C#异步操作等待窗体,异步多线程处理数据通用界面(frmThreadOperating)
C/S开发框架事务处理机制
【原创】WebApi开发框架:Token生成、Token缓存原理、Token验证、令牌机制与原理
CSFramework.WebApi系统安全保障机制
CSFramework.WebApi时钟定时作业异步处理程序(IThreadHandler接口)实现

热门标签
软件著作权登记证书 .NET .NET Reactor .NET5 .NET6 .NET7 .NET8 .NET9 .NETFramework APP AspNetCore AuthV3 Auth-软件授权注册系统 Axios B/S B/S开发框架 B/S框架 BSFramework Bug Bug记录 C#加密解密 C#源码 C/S CHATGPT CMS系统 CodeGenerator CSFramework.DB CSFramework.EF CSFramework.License CSFrameworkV1学习版 CSFrameworkV2标准版 CSFrameworkV3高级版 CSFrameworkV4企业版 CSFrameworkV5旗舰版 CSFrameworkV6.0 CSFrameworkV6.1 CSFrameworkV6旗舰版 DAL数据访问层 Database datalock DbFramework Demo教学 Demo实例 Demo下载 DevExpress教程 Docker Desktop DOM ECS服务器 EFCore EF框架 Element-UI EntityFramework ERP ES6 Excel FastReport GIT HR IDatabase IIS JavaScript LINQ MES MiniFramework MIS MySql NavBarControl NETCore Node.JS NPM OMS Oracle资料 ORM PaaS POS Promise API PSD RedGet Redis RSA SAP Schema SEO SEO文章 SQL SQLConnector SQLite SqlServer Swagger TMS系统 Token令牌 VS2022 VSCode VS升级 VUE WCF WebApi WebApi NETCore WebApi框架 WEB开发框架 Windows服务 Winform 开发框架 Winform 开发平台 WinFramework Workflow工作流 Workflow流程引擎 XtraReport 安装环境 版本区别 报表 备份还原 踩坑日记 操作手册 达梦数据库 代码生成器 迭代开发记录 功能介绍 国际化 基础资料窗体 架构设计 角色权限 开发sce 开发工具 开发技巧 开发教程 开发框架 开发平台 开发指南 客户案例 快速搭站系统 快速开发平台 框架升级 毛衫行业ERP 秘钥 密钥 权限设计 软件报价 软件测试报告 软件加壳 软件简介 软件开发框架 软件开发平台 软件开发文档 软件授权 软件授权注册系统 软件体系架构 软件下载 软件著作权登记证书 软著证书 三层架构 设计模式 生成代码 实用小技巧 视频下载 收钱音箱 数据锁 数据同步 微信小程序 未解决问题 文档下载 喜鹊ERP 喜鹊软件 系统对接 详细设计说明书 新功能 信创 行政区域数据库 需求分析 疑难杂症 蝇量级框架 蝇量框架 用户管理 用户开发手册 用户控件 在线支付 纸箱ERP 智能语音收款机 自定义窗体 自定义组件 自动升级程序
联系我们
联系电话:13923396219(微信同号)
电子邮箱:23404761@qq.com
站长微信二维码
微信二维码