AScript异步执行与await关键字

AScript是一个开源的C#动态脚本解析执行引擎,从1.2.5版本开始支持异步解析执行以及新增 await 和 @@CancellationToken 关键字。

一、异步解析执行

AScript提供了 Script.EvalAsync 异步方法,异步执行脚本,可设置 CancellationToken 参数。

AScript执行模式有解析执行和编译执行两种模式,这两种模式下的异步执行又有所不同:

1)解析执行模式:异步解析,异步执行,并且异步执行脚本中 await 后面的方法;

2)编译执行模式:异步解析,同步编译,同步执行,脚本中 await 后面的方法则会调用 Task.Wait() 或者 Task<T>.Result 来等待完成。

示例:

复制代码
1 var script = new Script();
2 // 100毫秒超时
3 var cts = new CancellationTokenSource(100);
4 // 异步读取脚本文件,异步解析脚本,异步执行脚本,如果超时100毫秒则报OperationCanceledException异常
5 var result = await script.EvalAsync(File.OpenRead("mycode.txt"), cancellationToken: cts.Token);
6 Console.WriteLine($"value:{result.Value}, type:{result.Type}");

二、await关键字

脚本中调用异步方法并等待完成,使用 await 关键字,当然也可以直接调用 Task.Wait() 或者 Task<T>.Result 来等待完成,为什么推荐使用 await 呢?

1)异步执行时,脚本中的方法也是异步的;

2)同步执行时,等效于 Task.Wait() 或者 Task<T>.Result ;

3)如果 await 后面的方法不是异步方法,则自动忽略 await ;

所以脚本中使用 await 关键字更省心更安全。

示例:

复制代码
 1 Func<int, int, Task<int>> sum = async (a, b) =>
 2 {
 3     await Task.Delay(1000);
 4     return a + b;
 5 };
 6 string s = @"
 7 var a = await sum(5, 10);
 8 await Task.Delay(500);
 9 a + 20
10 ";
11 var script = new Script();
12 script.Context.AddFunc("sum", sum);
13 // 异步执行
14 var result = await script.EvalAsync(s);
15 Assert.AreEqual(35, result.Value);
16 Assert.AreEqual(typeof(int), result.Type);
17 // 同步执行,等效于:var a = sum(5, 10).Result; Task.Delay(500).Wait(); a + 20
18 Assert.AreEqual(35, script.Eval(s));

await 关键字实现原理请查看源码 AScript.TokenHandlers.AwaitTokenHandler 和 AScript.Functions.AwaitFunction 。

三、@@CancellationToken关键字

1)异步执行时,值为 EvalAsync 方法参数中的 CancellationToken ;

2)同步执行时,值为 CancellationToken.None 。

示例:

复制代码
1 var script = new Script();
2 // 100毫秒超时
3 var cts = new CancellationTokenSource(100);
4 await Assert.ThrowsExceptionAsync<TaskCanceledException>(async () =>
5 {
6     await script.EvalAsync("await Task.Delay(1000, @@CancellationToken)", cancellationToken: cts.Token);
7 });

实现原理:

复制代码
 1 public class CancellationTokenHandler : ITokenHandler, IAsyncTokenHandler
 2 {
 3     public static readonly CancellationTokenHandler Instance = new CancellationTokenHandler();
 4 
 5     public void Build(DefaultSyntaxAnalyzer analyzer, TokenAnalyzingArgs e)
 6     {
 7         e.IsHandled = true;
 8         if (!e.Ignore)
 9         {
10             e.TreeBuilder.AddData(e.BuildContext, e.ScriptContext, e.Options, e.Control, CancellationToken.None);
11         }
12     }
13 
14     public async Task BuildAsync(DefaultSyntaxAnalyzer analyzer, TokenAnalyzingArgs e, CancellationToken cancellationToken)
15     {
16         e.IsHandled = true;
17         if (!e.Ignore)
18         {
19             e.TreeBuilder.AddData(e.BuildContext, e.ScriptContext, e.Options, e.Control, cancellationToken);
20         }
21     }
22 }

在 CSharpLang 中注册: AddTokenHandler("@@CancellationToken", CancellationTokenHandler.Instance);

四、总结

AScript提供了 EvalAsync 方法来异步读取、解析、执行脚本;

脚本中使用 await 和 @@CancellationToken 关键字来异步调用方法及传递取消令牌并等待结果。

开源地址:https://gitee.com/rockey627/AScript

相关推荐
hez20108 小时前
在 .NET 上构建超大托管数组
c#·.net·.net core·gc·clr
雨落倾城夏未凉6 天前
第四章c#方法-参数数组和可选参数(16)
后端·c#
唐青枫7 天前
线程不是越多越快:C#.NET Thread 生命周期、同步与后台工作线程实战
c#·.net
唐青枫8 天前
别只会反射:C#.NET Emit 动态生成代码实战详解
c#·.net
Caco_D8 天前
一行代码抓遍全网 20 个热榜!Aneiang.Pa 4.0 发布 — 极简 .NET 爬虫库
爬虫·.net
咕白m6258 天前
.NET 环境下 Word 超链接批量提取方案
c#·.net
用户91721561902118 天前
C# 通信协议增量解析:用状态机处理半包和粘包
c#
小码编匠9 天前
C# 工控上位机必备:数据转换工具类与十个核心模块
后端·c#·.net