Enum类型属性的RestSharp反序列化
本文关键字:RestSharp 反序列化 属性 类型 Enum | 更新日期: 2023-09-27 18:10:27
我有一个对象
var testTcc = new TrendingConfigurationConfigDto
{
TrendingConfigurationId =1,
ConfigId = 1,
DeviceId = 1,
Selected = true,
YAxisPosition = YAxisPosition.Left,
Order = 1,
Color = "#ffffff",
Configuration = new BO.Shared.Dtos.List.ConfigurationListDto
{
Id = 1,
Name = "configuration",
Alias = "configuationAlias",
EnableEdit = true,
IsBusinessItem = true
},
Device = new BO.Shared.Dtos.List.DeviceListDto
{
Id = 1,
Name = "Device"
}
};
当我将它序列化为json
时var jsonTcc = SimpleJson.SerializeObject(testTcc);
它返回包含YAxisPosition = 1的json对象的字符串,当我尝试使用
对其进行反序列化时testTcc = SimpleJson.DeserializeObject<TrendingConfigurationConfigDto>(jsonTcc);
它给出了一个异常系统。InvalidCastException,提示"指定的类型转换无效"。
我尝试将json字符串中的YAxisPosition值更改为字符串"1"或"Left",它总是给我同样的错误,直到我从json字符串中删除属性YAxisPosition。
我可能遗漏了一些东西(枚举属性上的属性或类似的东西)。
请帮助我找到一种方法,使我可以序列化和反序列化一个对象,其中包含Enum类型的属性,使用RestSharp。
注意:我使用NewtonSoft尝试了成功的序列化和反序列化。但是我不希望我的Web API客户端依赖于nettwonsoft,因为我已经在使用RestSharp。
RestSharp删除了JSON。. NET支持v103.0。默认的Json序列化器不再与Json.NET兼容。如果您想继续使用JSON,有几个选项。NET并保持向后兼容性。除此之外,还有JSON。.NET有更多的功能,并且可以解决您使用RestSharp现在依赖的基本.NET序列化器的问题。
另外,你可以使用[EnumMember]属性在反序列化过程中定义自定义映射。
选项1:实现使用JSON的自定义序列化器。净
使用Json。. NET进行序列化,复制以下代码:
/// <summary>
/// Default JSON serializer for request bodies
/// Doesn't currently use the SerializeAs attribute, defers to Newtonsoft's attributes
/// </summary>
public class JsonNetSerializer : ISerializer
{
private readonly Newtonsoft.Json.JsonSerializer _serializer;
/// <summary>
/// Default serializer
/// </summary>
public JsonSerializer() {
ContentType = "application/json";
_serializer = new Newtonsoft.Json.JsonSerializer {
MissingMemberHandling = MissingMemberHandling.Ignore,
NullValueHandling = NullValueHandling.Include,
DefaultValueHandling = DefaultValueHandling.Include
};
}
/// <summary>
/// Default serializer with overload for allowing custom Json.NET settings
/// </summary>
public JsonSerializer(Newtonsoft.Json.JsonSerializer serializer){
ContentType = "application/json";
_serializer = serializer;
}
/// <summary>
/// Serialize the object as JSON
/// </summary>
/// <param name="obj">Object to serialize</param>
/// <returns>JSON as String</returns>
public string Serialize(object obj) {
using (var stringWriter = new StringWriter()) {
using (var jsonTextWriter = new JsonTextWriter(stringWriter)) {
jsonTextWriter.Formatting = Formatting.Indented;
jsonTextWriter.QuoteChar = '"';
_serializer.Serialize(jsonTextWriter, obj);
var result = stringWriter.ToString();
return result;
}
}
}
/// <summary>
/// Unused for JSON Serialization
/// </summary>
public string DateFormat { get; set; }
/// <summary>
/// Unused for JSON Serialization
/// </summary>
public string RootElement { get; set; }
/// <summary>
/// Unused for JSON Serialization
/// </summary>
public string Namespace { get; set; }
/// <summary>
/// Content type for serialized content
/// </summary>
public string ContentType { get; set; }
}
并注册到您的客户端:
var client = new RestClient();
client.JsonSerializer = new JsonNetSerializer();
选项2:使用nuget包来使用JSON。净
而不是做所有这些,并有一个自定义JSON序列化器分布在您的项目是使用这个nuget包:https://www.nuget.org/packages/RestSharp.Newtonsoft.Json。它允许您使用继承的RestRequest对象,默认使用Newtonsoft。var request = new RestSharp.Newtonsoft.Json.RestRequest(); // Uses JSON.NET
另一个选项是在每个请求上设置它,像这样:
var request = new RestRequest();
request.JsonSerializer = new NewtonsoftJsonSerializer();
免责声明:在我的项目中有一个自定义序列化器后,我创建了这个项目。我创建这个是为了保持整洁,并希望帮助那些希望向后兼容v103之前工作的RestSharp代码的人。
找到解决方案:
private IRestClient GetRestClient()
{
return new RestClient(url)
.AddDefaultHeader("Authorization", $"Bearer {token.AccessToken}")
.AddDefaultHeader("Accept", "*/*")
.AddDefaultHeader("Accept-Encoding", "gzip, deflate, br")
.AddDefaultHeader("Connection", "close")
.UseSystemTextJson(new JsonSerializerOptions
{
Converters = { new JsonStringEnumConverter() }
});
}
。指示RestSharp使用System.Text.Json序列化器,然后指示序列化器使用JsonStringEnumConverter类来序列化/反序列化枚举。
我在PocoJsonSerializerStrategy
的帮助下得到了它的工作。RestSharp允许您指定自己的序列化/反序列化策略,因此我创建了自己的策略来处理枚举:
public class HandleEnumsJsonSerializerStrategy : PocoJsonSerializerStrategy
{
public override object DeserializeObject(object value, Type type)
{
if (type.IsEnum)
return Enum.Parse(type, (string)value);
else
return base.DeserializeObject(value, type);
}
}
现在您可以将该类的对象传递给SimpleJson.DeserializeObject
调用,就像这样,您的枚举被优雅地处理:
SimpleJson.DeserializeObject<JsonObject>(Response.Content, Strategy);
如果您正在使用枚举类模式,您可以使用JsonConverter
方法。这意味着您需要创建一个自定义转换器,并在需要序列化的属性上使用JsonConverter
属性。示例如下:
using Newtonsoft.Json;
using System;
namespace MyAwesomeAPI
{
public class EnumerationConverter<TEnum> : JsonConverter
where TEnum : Enumeration
{
public override bool CanConvert(Type objectType) => objectType == typeof(TEnum);
public override bool CanRead { get => true; }
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
var uknown = Enumeration.FromDisplayName<TEnum>("Unknown");
if (reader.TokenType == JsonToken.Null)
{
return null;
}
if (reader.TokenType == JsonToken.String)
{
var enumText = reader.Value?.ToString();
if (string.IsNullOrEmpty(enumText))
{
return uknown;
}
try
{
return Enumeration.FromDisplayName<TEnum>(enumText);
}
catch (Exception)
{
return uknown;
}
}
return uknown;
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
writer.WriteValue(value.ToString());
}
}
}
我们的enum看起来像:
public class CardType : Enumeration
{
public static CardType Unknown = new(0, nameof(Unknown));
public static CardType Amex = new(1, nameof(Amex));
public static CardType Visa = new(2, nameof(Visa));
public static CardType MasterCard = new(3, nameof(MasterCard));
public CardType(int id, string name) : base(id, name)
{
}
}
最后是将属性应用于属性
public class CreditCardResponse
{
[JsonProperty("card_number")]
public string CardNumber { get; set; }
[JsonProperty("card_type")]
[JsonConverter(typeof(EnumerationConverter<CardType>))]
public CardType CardType { get; set; }
}