【问题标题】:C# Newtonsoft Deserialize Custom Object with Claims?C# Newtonsoft 使用声明反序列化自定义对象?
【发布时间】:2020-12-17 10:32:13
【问题描述】:

我有以下简单的 POCO:

  public class ApiKey
  {
    public ApiKey(string key, string owner, List<Claim> claims = null)
    {
      Key = key;
      OwnerName = owner;
      Claims = claims ?? new List<Claim>();
    }

    public string Key { get; }
    public string OwnerName { get; }
    public IReadOnlyCollection<Claim> Claims { get; set; }
  }

我可以用一个声明创建这个对象的一个​​实例,然后用 Newtonsoft 序列化它:

JsonConvert.SerializeObject(key)

然后像这样得到一个序列化的 ApiKey:

"{\"Key\":\"94a5b81f-9837-4c5f-9821-3ebaedc6435d\",\"OwnerName\":null,\"Claims\":[{\"Issuer\":\"LOCAL AUTHORITY\",\"OriginalIssuer\":\"LOCAL AUTHORITY\",\"Properties\":{},\"Subject\":null,\"Type\":\"AdminClaim\",\"Value\":\"AdminClaim\",\"ValueType\":\"http://www.w3.org/2001/XMLSchema#string\"}]}"

但是,如果尝试像这样使用 Newtonsoft 反序列化该字符串:

JsonConvert.DeserializeObject<ApiKey>(serialized_key);

我收到以下错误:

Newtonsoft.Json.JsonSerializationException: Unable to find a constructor to use for type System.Security.Claims.Claim. A class should either have a default constructor, one constructor with arguments or a constructor marked with the JsonConstructor attribute. Path 'Claims[0].Issuer', line 1, position 83.
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateNewObject(JsonReader reader, JsonObjectContract objectContract, JsonProperty containerMember, JsonProperty containerProperty, String id, Boolean& createdFromNonDefaultCreator)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateObject(JsonReader reader, Type objectType, JsonContract contract, JsonProperty member, JsonContainerContract containerContract, JsonProperty containerMember, Object existingValue)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateValueInternal(JsonReader reader, Type objectType, JsonContract contract, JsonProperty member, JsonContainerContract containerContract, JsonProperty containerMember, Object existingValue)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.PopulateList(IList list, JsonReader reader, JsonArrayContract contract, JsonProperty containerProperty, String id)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateList(JsonReader reader, Type objectType, JsonContract contract, JsonProperty member, Object existingValue, String id)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateValueInternal(JsonReader reader, Type objectType, JsonContract contract, JsonProperty member, JsonContainerContract containerContract, JsonProperty containerMember, Object existingValue)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.ResolvePropertyAndCreatorValues(JsonObjectContract contract, JsonProperty containerProperty, JsonReader reader, Type objectType)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateObjectUsingCreatorWithParameters(JsonReader reader, JsonObjectContract contract, JsonProperty containerProperty, ObjectConstructor`1 creator, String id)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateObject(JsonReader reader, Type objectType, JsonContract contract, JsonProperty member, JsonContainerContract containerContract, JsonProperty containerMember, Object existingValue)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.CreateValueInternal(JsonReader reader, Type objectType, JsonContract contract, JsonProperty member, JsonContainerContract containerContract, JsonProperty containerMember, Object existingValue)
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.Deserialize(JsonReader reader, Type objectType, Boolean checkAdditionalContent)
   at Newtonsoft.Json.JsonSerializer.DeserializeInternal(JsonReader reader, Type objectType)
   at Newtonsoft.Json.JsonConvert.DeserializeObject(String value, Type type, JsonSerializerSettings settings)
   at Newtonsoft.Json.JsonConvert.DeserializeObject[T](String value, JsonSerializerSettings settings)

当我希望取回我最初序列化的原始对象时。为了能够反序列化这个对象,我缺少什么?

谢谢!

编辑:Claim 对象来自 System.Security.Claims.Claim。

编辑 2:我希望能够序列化/反序列化整个 ApiKey 类,而不仅仅是声明部分。

编辑 3:我无法修改 ApiKey 类。

【问题讨论】:

  • 错误很明显,对吧?仅仅因为一个类在没有合适的构造函数时是可序列化的,并不意味着没有合适的构造函数它是可反序列化的。这不是简单地复制一块内存的问题。
  • 问题似乎是构造函数中的 Claim 类。如果您阅读了错误消息的第一行,则表明它
  • 我正在使用 System.Security.Claims.Claim 类。知道如何让它反序列化吗?

标签: c# .net asp.net-mvc asp.net-core .net-core


【解决方案1】:

您必须编写一个自定义转换器,此答案中有一个 Claim 类的示例:https://stackoverflow.com/a/28155770/6881299

【讨论】:

  • 您必须使用 [JsonProperty(ItemConverterType = typeof(ClaimConverter))] 属性标记您的 IReadonlyCollection 属性。
  • 另一种选择是使用 IdentityServer4 库中包含的 ClaimConverter,如下所述:stackoverflow.com/a/54645939/10069673
【解决方案2】:

问题是Claim 类没有任何公共构造函数。最简单的解决方法是创建自己的 Claim 类并将其反序列化:

class MyClaim : Claim {
    public MyClaim(string type, string value, string valueType, string issuer, string originalIssuer):
        base(type, value, valueType, issuer, originalIssuer){}
}

Claim claim = JsonConvert.DeserializeObject<MyClaim>(json);

【讨论】:

    猜你喜欢
    • 2017-05-26
    • 1970-01-01
    • 1970-01-01
    • 2015-10-29
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2022-01-25
    • 1970-01-01
    相关资源
    最近更新 更多