【问题标题】:Import deserialized System.Text.Json data into current type [duplicate]将反序列化的 System.Text.Json 数据导入当前类型 [重复]
【发布时间】:2022-03-09 03:36:26
【问题描述】:

我想从作为反序列化目标的类中导入 json 数据。 System.Text.Json 是否可以在没有额外映射的情况下实现这一点?理想情况下,我会使用“this”而不是泛型类型参数。我知道这是不可能的,但有类似的选择吗?这是我的有效测试代码,因为它创建数据对象只是为了将其映射到属性。理想情况下,我不需要两次实例化“测试”。

public class Test
{
    public string? Bar { get; set; }
    
    public void ImportJson(string payload)
    {
        var data = System.Text.Json.JsonSerializer.Deserialize<Test>(payload);
        Bar = data?.Bar; // Don't want to map
    }
}

string foo = "{ \"Bar\": \"baz\" }";
var t = new Test();
t.ImportJson(foo);
Console.WriteLine(t.Bar);

【问题讨论】:

  • 不,这是不可能的。 Deserialize 只能创建新实例,没有可以填充现有实例的重载。因此,提供ImportJson 作为实例方法几乎没有意义,只需将其设为static Test FromJson(string payload) 或类似方法即可。 (当然,当您拥有它时,它只是 Deserialize 的一个微不足道的包装器,因此可能没有添加任何值。)
  • 您正在寻找Populate() 方法,但与Newtonsoft 不同的是,有no such method built in to System.Text.Json。请参阅:.Net Core 3.0 JsonSerializer populate existing object。事实上,这看起来是重复的,同意吗?
  • 同意,这个问题是重复的,我没有发现它缺少知道Json.net“Populate()”方法。

标签: c# json json-deserialization system.text.json


【解决方案1】:

你可以试试这样的

    string foo = "{ \"Bar\": \"baz\" }";
    var t = new Test();
    t.Deserialize(foo);

    Console.WriteLine(t.Instance.Bar);

public static class Util
{
    public static void Deserialize<T>(this T obj, string json) where T : IImportJson<T>
    {
        obj.Instance=System.Text.Json.JsonSerializer.Deserialize<T>(json);
    }
}

public class Test : ImportJson<Test>
{
    public string? Bar { get; set;}
    
}

public interface IImportJson<T>
{
    public T Instance { get; set; }
}
public class ImportJson<T>: IImportJson<T>
{
    public T Instance { get; set; }
}

如果类没有很多属性,也可以这样

public interface IImportJson<T>
{
    public void ImportJson (T obj);
}
public class Test : IImportJson<Test>
{
    public string? Bar { get; set; }
    
    public void ImportJson(Test test)
    {
        Bar=test.Bar;
    }
}

【讨论】:

  • 这仍然需要一个额外的实例,但是看到这个让我意识到我应该重新组织我的代码并使用反序列化器来创建唯一的实例,类似于 JsonSerializer.Deserialize(foo, type ) 因为在我的情况下类型是动态的。
  • 额外实例是什么意思?在这个演示案例中,您只需分配一个属性,不保留其他任何内容。
  • 我将拥有“Test”和封装的“Test.Instance”。用 JsonSerializer 实例化的“new”和“Test.Instance”实例化的“Test”。但是,它确实满足了我最初的要求。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2020-04-06
  • 1970-01-01
  • 1970-01-01
  • 2016-04-06
  • 1970-01-01
  • 2021-07-17
  • 2020-06-23
相关资源
最近更新 更多