您可以通过自定义进行 *** 作
ContractResolver。解析器可以寻找一个自定义属性,该属性将表明您希望JSON属性的名称基于可枚举项目的类。如果项目类具有指定其复数名称的另一个属性,则该名称将用作可枚举属性,否则,项目类名称本身将被复数并用作可枚举属性名称。以下是您需要的代码。
首先让我们定义一些自定义属性:
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(看起来像您一样),则可以通过类的
Register方法
WebApiConfig(在
App_Start文件夹中)将自定义解析器安装到管道中。
JsonSerializerSettings settings = config.Formatters.JsonFormatter.SerializerSettings;settings.ContractResolver = new CustomResolver();
另一种方法
另一种可能的方法是使用自定义
JsonConverter来
PagedData专门处理类的序列化,而不是使用上面介绍的更通用的“解析器+属性”方法。转换器方法要求您的
PagedData类上还有另一个属性,该属性指定要用于可枚举
Data属性的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类中添加一个名为string的字符串属性(如果愿意,可以是私有的),然后
[JsonConverter]在该类中添加一个属性以将其绑定到转换器:
[JsonConverter(typeof(PagedDataConverter))]public class PagedData<T>{ private string DataPropertyName { get; set; } public IEnumerable<T> Data { get; private set; } ...}
就是这样。只要设置了
DataPropertyName属性,转换器将在序列化时将其拾取。
小提琴:https :
//dotnetfiddle.net/8E8fEE
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)