Ответ 1
Вы можете сделать это с помощью настраиваемого 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; }
...
}
И украсьте классы DTO атрибутом [JsonPluralName]
:
[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);
Fiddle: https://dotnetfiddle.net/GqKBnx
Если вы используете веб-API (похоже, что вы есть), вы можете установить настраиваемый преобразователь в конвейер с помощью метода Register
класса WebApiConfig
(в папке App_Start
).
JsonSerializerSettings settings = config.Formatters.JsonFormatter.SerializerSettings;
settings.ContractResolver = new CustomResolver();
Другой подход
В другом возможном подходе используется пользовательский JsonConverter
для обработки сериализации класса PagedData
, а вместо этого используется более общий подход "resolver + attributes", представленный выше. Подход конвертера требует наличия другого свойства в вашем классе PagedData
, который указывает имя JSON, которое будет использоваться для перечислимого свойства Data
. Вы можете передать это имя в конструкторе 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();
}
}
Чтобы использовать этот конвертер, сначала добавьте свойство string, называемое DataPropertyName
, в ваш класс PagedData
(он может быть приватным, если хотите), а затем добавьте к классу атрибут [JsonConverter]
для привязки к конвертеру:
[JsonConverter(typeof(PagedDataConverter))]
public class PagedData<T>
{
private string DataPropertyName { get; set; }
public IEnumerable<T> Data { get; private set; }
...
}
И что это. Пока вы устанавливаете свойство DataPropertyName
, он будет подбираться преобразователем при сериализации.
Fiddle: https://dotnetfiddle.net/8E8fEE