首页 > 解决方案 > 我在这里制造泄漏吗?

问题描述

我正在使用 NETCore 3.0 的 System.Text.Json 命名空间中的新 JsonSerializer 来反序列化 Json 文档,如下所示:

var result = JsonSerializer.Deserialize<Response>(json, options);

响应定义为:

public class Response
{
    public string Foo { get; set; }
    public JsonElement Bar { get; set; }
}

JsonDocument实现 IDisposable的事实让我想知道,如果保留对Bar可以包含在 JsonDocument 中的元素 () 的引用,是否会造成内存泄漏?

请注意,通常我会避免将数据存储为像这样的“变体”类型。不幸的是,Bar属性值的结构在编译时是未知的。

我的怀疑源于 System.Text.Json 宣传的懒惰评估的优势,我不确定这是否涉及延迟 I/O。

标签: c#jsonidisposable.net-core-3.0system.text.json

解决方案


从对来源的简要调查(https://github.com/dotnet/corefx/blob/master/src/System.Text.Json/src/System/Text/Json/Document/JsonDocument.cs)看来,JsonDocument Dispose将“租用”字节返回到共享数组池并进行一些常规清理。JsonDocument 的某些实例被标记为不可丢弃,在这种情况下 Dispose 不会做任何事情。您可以使用反射检查您的实例的此标志 - 如果您的实例没有将内部 IsDisposable 标志设置为 true,则无需担心,因为 Dispose 无论如何都不会执行任何操作。

我认为在正常情况下,JsonDocument 解析器应该自行清理,并且解析器完成后应该没有租用的字节或内部数据。

不依赖特定的实现总是安全的,因为它可能会更改并仅存储对所需元素的引用。您可能应该将 JSON 元素重新映射到您的模型,我认为这就是 JSON 反序列化的全部目的

快速测试:

        var parentField = result.Bar.GetType().GetMember("_parent", MemberTypes.Field, BindingFlags.Instance | BindingFlags.NonPublic)[0] as FieldInfo;
        var parentDocument = parentField.GetValue(result.Bar);

        var isDisposableProperty = parentDocument.GetType().GetProperty("IsDisposable", BindingFlags.Instance | BindingFlags.NonPublic) as PropertyInfo;
        Console.WriteLine(isDisposableProperty.GetValue(parentDocument)); // false

证明 JsonElement 持有的 JsonDocument 的实例不是一次性的。


推荐阅读