一尘不染

使用System.Text.Json修改JSON文件

json

我知道您可以使用Newtonsoft轻松地做到这一点。但是,当我使用.NET Core
3.0时,我正在尝试使用新方法与JSON文件进行交互,即,System.Text.Json并且我拒绝相信我要做的一切都那么困难!

我的应用程序需要列出尚未添加到我的数据库中的用户。为了获取所有用户的完整列表,该应用程序从Web
API检索JSON字符串。现在,我需要循环浏览这些用户中的每一个,并检查是否已将它们添加到我的应用程序中,然后再将新的JSON列表返回到我的视图,以便它可以向最终用户显示新的潜在用户。

由于我最终会在流程结束时返回另一个JSON,因此我特别不想打扰将其反序列化为模型。请注意,来自API的数据结构 可能会 发生变化,但是它将 始终
具有一个密钥,我可以将其与数据库记录进行比较。

我的代码当前如下所示:

using (WebClient wc = new WebClient())
{
    var rawJsonDownload = wc.DownloadString("WEB API CALL");
    var users =  JsonSerializer.Deserialize<List<UserObject>>(rawJsonDownload);

    foreach (var user in users.ToList())
    {
        //Check if User is new
        if (CHECKS)
        {
            users.Remove(user);
        }
    }

    return Json(users); 
}

这似乎是一个 很大 的篮球,以实现的东西,这将是与Newtonsoft相当琐碎通过跳跃。

有人可以建议我采取更好的方法UserObject吗?理想情况下,不需要这样做吗?


阅读 1008

收藏
2020-07-27

共1个答案

一尘不染

您的问题是您想检索,过滤和传递一些JSON,而无需为该JSON定义完整的数据模型。使用Json.NET,您可以为此使用LINQ to
JSON
。您的问题是,
这目前可以轻松解决System.Text.Json吗?

从.NET Core 3.0开始,System.Text.Json由于以下原因,此操作无法如此轻松地完成:

  1. JsonDocument,对应于JToken或的类型XDocument为只读。它只能用于 检查 JSON值,不能用于修改或创建JSON值。

当前有一个未解决的问题 Writable Json
DOM#39922对此进行了
跟踪。

  1. System.Text.Json不支持JSONPath,这在此类应用程序中通常很方便。

当前存在一个未解决的问题, 将JsonPath支持添加到JsonDocument /
JsonElement#41537,以对此进行
跟踪。

话虽如此,假设您有以下JSON:

[
  {
    "id": 1,
    "name": "name 1",
    "address": {
      "Line1": "line 1",
      "Line2": "line 2"
    },
    // More properties omitted
  }
  //, Other array entries omitted
]

以及Predicate<long>shouldSkip与您的问题id相对应的某种过滤方法,该方法指示是否应返回不包含特定条目的条目CHECKS。您有什么选择?

您可以使用JsonDocument并返回一些经过过滤的JsonElement节点集。如果过滤逻辑非常简单,并且您不需要以任何其他方式修改JSON,则这很有意义。注意,JsonDocument是一次性的,并且实际上必须需要被设置为
最小化在高的使用场景的垃圾收集器(GC)的影响 ,根据该文档。因此,为了返回a,JsonElement您必须对其进行克隆

以下代码显示了此示例:

using var usersDocument = JsonDocument.Parse(rawJsonDownload);
var users = usersDocument.RootElement.EnumerateArray()
    .Where(e => !shouldSkip(e.GetProperty("id").GetInt64()))
    .Select(e => e.Clone())
    .ToList();

return Json(users);

样机小提琴#1 在这里

您可以创建一个 部分
数据模型,该模型仅反序列化过滤所需的属性,而其余JSON绑定到一个[JsonExtensionDataAttribute]属性。

这应该使您无需硬编码整个数据模型即可实施必要的过滤。

为此,请定义以下模型:

public class UserObject
{
    [JsonPropertyName("id")]
    public long Id { get; set; }

    [System.Text.Json.Serialization.JsonExtensionDataAttribute]
    public IDictionary<string, object> ExtensionData { get; set; }
}

并反序列化和过滤如下:

var users = JsonSerializer.Deserialize<List<UserObject>>(rawJsonDownload);
users.RemoveAll(u => shouldSkip(u.Id));

return Json(users);

这种方法确保了与过滤相关的属性可以适当地反序列化,而无需对JSON的其余部分做任何假设。尽管这并不像使用LINQ to
JSON那样容易,但是总代码复杂度受筛选检查的复杂度限制,而不是JSON的复杂度。实际上,我的观点是,实际上,这种方法比纯JsonDocument方法更容易使用,因为如果以后需要,它可以更轻松地注入对JSON的修改。

样机小提琴#2 在这里

没有你选择事情 ,你可能会考虑抛弃WebClientHttpClient,并使用async反序列化。例如:

var httpClient = new HttpClient();
using var usersDocument = await JsonDocument.ParseAsync(await httpClient.GetStreamAsync("WEB API CALL"));

要么

var users = await JsonSerializer.DeserializeAsync<List<UserObject>>(await httpClient.GetStreamAsync("WEB API CALL"));

您还需要将您的API方法转换async为。

2020-07-27