Rx 库是否省略了它创建的 CancellationTokenSources 的处理?

如何解决Rx 库是否省略了它创建的 CancellationTokenSources 的处理?

Rx 库包含接受 lambda 参数的运算符,其中一些 lambda 表达式带有 CancellationToken,该 FromAsync 由库本身控制。这些运算符的一些示例是 CancellationTokenSourceStartAsyncCreate

// Converts an asynchronous action into an observable sequence. Each subscription
// to the resulting sequence causes the action to be started. The CancellationToken
// passed to the asynchronous action is tied to the observable sequence's subscription
// that triggered the action's invocation and can be used for best-effort cancellation.
public static IObservable<Unit> FromAsync(Func<CancellationToken,Task> actionAsync);

我的印象是,Rx 库在管理显然必须在幕后创建的 insists strongly 的生命周期方面做得很好,但我不再那么确定了。让我们首先声明应该处理 CancellationTokenSource 的文档 here

这种类型实现了 Idisposable 接口。当您使用完该类型的实例后,您应该直接或间接地处理它。要直接处理该类型,请在 dispose/try 块中调用catch 方法。要间接处理它,请使用诸如 using(在 C# 中)或 Using(在 Visual Basic 中)之类的语言结构。

也来自_source

在释放对 dispose 的最后一次引用之前,始终调用 CancellationTokenSource。否则,在垃圾收集器调用 CancellationTokenSource 对象的 Finalize 方法之前,它使用的资源不会被释放。

我做了下面的实验来检验我的假设。它使用反射分别读取 CancellationTokenCancellationTokenSource 类型的私有字段 _disposedTry it on Fiddle (.NET 5)。

CancellationToken capturedToken = default;
var subscription = Observable.FromAsync(async token =>
{
    capturedToken = token;
    token.Register(() => Console.WriteLine("Token canceled"));
    await Task.Delay(Timeout.Infinite,token);
})
.takeuntil(Observable.Timer(TimeSpan.FromMilliseconds(500)))
.Finally(() => Console.WriteLine("The observable was terminated"))
.Subscribe();

Thread.Sleep(1000);

var cts = (CancellationTokenSource)(typeof(CancellationToken)
    .GetField("_source",BindingFlags.NonPublic | BindingFlags.Instance)
    .GetValue(capturedToken));
bool disposed = (bool)(typeof(CancellationTokenSource)
    .GetField("_disposed",BindingFlags.NonPublic | BindingFlags.Instance)
    .GetValue(cts));
Console.WriteLine($"IsCancellationRequested: {cts.IsCancellationRequested}");
Console.WriteLine($"Isdisposed: {disposed}");

输出

Token canceled
The observable was terminated
IsCancellationRequested: True
Isdisposed: False

{{3}}(.NET Framework 版本,具有不同命名的私有字段)

在异步操作取消且可观察对象终止后半秒检查捕获的 CancellationToken_disposed 字段的值为 false,表示尚未调用关联的 disposeCancellationTokenSource 方法。我做错了什么,还是 Rx 库确实忽略了处理它创建的 CancellationTokenSource

.NET 5.0.1、System.Reactive 5.0.0、C# 9

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?