例如,如果我正在返回分页用户列表,Data属性应该被序列化为“用户”,如果我返回的项目列表,应该称为“项目”等.
是这样可能的:
public class PagedData { [JsonProperty(PropertyName = "Set from constructor")]?? public IEnumerable<T> Data { get; private set; } public int Count { get; private set; } public int CurrentPage { get; private set; } public int Offset { get; private set; } public int RowsPerPage { get; private set; } public int? PrevIoUsPage { get; private set; } public int? NextPage { get; private set; } }
编辑:
我想要控制这个功能,例如传递名字,以便尽可能使用.如果我的类被称为UserDTO,我仍然希望序列化属性被称为用户,而不是UserDTO.
例
var usersPagedData = new PagedData("Users",params...);
解决方法
public class JsonPropertyNameBasedOnItemClassAttribute : Attribute { } public class JsonPluralNameAttribute : Attribute { public string PluralName { get; set; } public JsonPluralNameAttribute(string pluralName) { PluralName = pluralName; } }
然后解析器:
public class CustomResolver : DefaultContractResolver { protected override JsonProperty CreateProperty(MemberInfo member,MemberSerialization memberSerialization) { JsonProperty prop = base.CreateProperty(member,memberSerialization); if (prop.PropertyType.IsGenericType && member.GetCustomAttribute<JsonPropertyNameBasedOnItemClassAttribute>() != null) { Type itemType = prop.PropertyType.GetGenericArguments().First(); JsonPluralNameAttribute att = itemType.GetCustomAttribute<JsonPluralNameAttribute>(); prop.PropertyName = att != null ? att.PluralName : Pluralize(itemType.Name); } return prop; } protected string Pluralize(string name) { if (name.EndsWith("y") && !name.EndsWith("ay") && !name.EndsWith("ey") && !name.EndsWith("oy") && !name.EndsWith("uy")) return name.Substring(0,name.Length - 1) + "ies"; if (name.EndsWith("s")) return name + "es"; return name + "s"; } }
现在,您可以在PagedData< T>中修饰可变命名的属性.类与[JsonPropertyNameBasedOnItemClass]属性:
public class PagedData<T> { [JsonPropertyNameBasedOnItemClass] public IEnumerable<T> Data { get; private set; } ... }
并使用[JsonPluralName]属性来装饰你的DTO类:
[JsonPluralName("Users")] public class UserDTO { ... } [JsonPluralName("Items")] public class ItemDTO { ... }
最后,要序列化,创建JsonSerializerSettings的一个实例,设置ContractResolver属性,并将设置传递给JsonConvert.SerializeObject,如下所示:
JsonSerializerSettings settings = new JsonSerializerSettings { ContractResolver = new CustomResolver() }; string json = JsonConvert.SerializeObject(pagedData,settings);
小提琴:https://dotnetfiddle.net/GqKBnx
如果您使用的是Web API(如您所愿),则可以通过WebApiConfig类的注册方法(在App_Start文件夹中)将自定义解析器安装到管道中.
JsonSerializerSettings settings = config.Formatters.JsonFormatter.SerializerSettings; settings.ContractResolver = new CustomResolver();
另一种方法
另一种可能的方法使用自定义JsonConverter来处理PagedData类的序列化,而不是使用上面提出的更一般的“解析器属性”方法.转换器方法要求您的PagedData类上有另一个属性,它指定要用于枚举数据属性的JSON名称.您可以在PagedData构造函数中传递此名称,也可以将其单独设置,只要在序列化时间之前执行此操作即可.转换器将寻找该名称并在为可枚举属性编写JSON时使用该名称.
以下是转换器的代码:
public class PagedDataConverter : JsonConverter { public override bool CanConvert(Type objectType) { return objectType.IsGenericType && objectType.GetGenericTypeDefinition() == typeof(PagedData<>); } public override void WriteJson(JsonWriter writer,object value,JsonSerializer serializer) { Type type = value.GetType(); var bindingFlags = BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public; string dataPropertyName = (string)type.GetProperty("DataPropertyName",bindingFlags).GetValue(value); if (string.IsNullOrEmpty(dataPropertyName)) { dataPropertyName = "Data"; } JObject jo = new JObject(); jo.Add(dataPropertyName,JArray.FromObject(type.GetProperty("Data").GetValue(value))); foreach (PropertyInfo prop in type.GetProperties().Where(p => !p.Name.StartsWith("Data"))) { jo.Add(prop.Name,new JValue(prop.GetValue(value))); } jo.WriteTo(writer); } public override bool CanRead { get { return false; } } public override object ReadJson(JsonReader reader,Type objectType,object existingValue,JsonSerializer serializer) { throw new NotImplementedException(); } }
要使用此转换器,首先添加一个名为DataPropertyName的字符串属性到您的PagedData类(如果您愿意,它可以是私有的),然后添加一个[JsonConverter]属性到该类以将其绑定到转换器:
[JsonConverter(typeof(PagedDataConverter))] public class PagedData<T> { private string DataPropertyName { get; set; } public IEnumerable<T> Data { get; private set; } ... }
就是这样只要你设置了DataPropertyName属性,它将被转换器在序列化时被拾取.