如何从 JSON 序列化中排除属性
本文关键字:排除 属性 序列化 JSON | 更新日期: 2023-09-27 18:31:18
>我有一个DTO类,我序列化了它
Json.Serialize(MyClass)
如何排除其公共属性?
(它必须是公开的,因为我在其他地方的代码中使用它)
使用的是 Json.Net 属性[JsonIgnore]
则在序列化或反序列化时将简单地忽略字段/属性。
public class Car
{
// included in JSON
public string Model { get; set; }
public DateTime Year { get; set; }
public List<string> Features { get; set; }
// ignored
[JsonIgnore]
public DateTime LastModified { get; set; }
}
或者,可以使用 DataContract 和 DataMember 属性有选择地序列化/反序列化属性/字段。
[DataContract]
public class Computer
{
// included in JSON
[DataMember]
public string Name { get; set; }
[DataMember]
public decimal SalePrice { get; set; }
// ignored
public string Manufacture { get; set; }
public int StockCount { get; set; }
public decimal WholeSalePrice { get; set; }
public DateTime NextShipmentDate { get; set; }
}
有关更多详细信息,请参阅 http://james.newtonking.com/archive/2009/10/23/efficient-json-with-json-net-reducing-serialized-json-size
如果在 .NET 框架中使用System.Web.Script.Serialization
,则可以在不应序列化的成员上放置ScriptIgnore
属性。请参阅此处的示例:
考虑以下(简化)情况:
public class User { public int Id { get; set; } public string Name { get; set; } [ScriptIgnore] public bool IsComplete { get { return Id > 0 && !string.IsNullOrEmpty(Name); } } }
在这种情况下,将仅序列化 Id 和 Name 属性,因此生成的 JSON 对象如下所示:
{ Id: 3, Name: 'Test User' }
不要忘记添加对" System.Web.Extensions
"的引用以使此操作正常工作
编辑 2023-06-29:更新了答案并添加了有关 .NET 核心和 System.Text.Json 的信息
如果您不想使用某些属性修饰属性,或者您无权访问该类,或者如果您想决定在运行时序列化什么,请按以下步骤操作:
1. 在牛顿软件.Json
Newtonsoft解决方案非常简单:
//short helper class to ignore some properties from serialization
public class IgnorePropertiesResolver : DefaultContractResolver
{
private readonly HashSet<string> ignoreProps;
public IgnorePropertiesResolver(IEnumerable<string> propNamesToIgnore)
{
this.ignoreProps = new HashSet<string>(propNamesToIgnore);
}
protected override JsonProperty CreateProperty(MemberInfo member, MemberSerialization memberSerialization)
{
JsonProperty property = base.CreateProperty(member, memberSerialization);
if (this.ignoreProps.Contains(property.PropertyName))
{
property.ShouldSerialize = _ => false;
}
return property;
}
}
用法
JsonConvert.SerializeObject(YourObject, new JsonSerializerSettings()
{ ContractResolver = new IgnorePropertiesResolver(new[] { "Prop1", "Prop2" }) });
如果您决定使用此答案,请确保缓存ContractResolver
对象,否则性能可能会受到影响。
我在这里发布了代码,以防有人想添加任何内容:https://github.com/jitbit/JsonIgnoreProps
2. 在System.Text.Json
.NET core默认使用System.Text.Json
,它更快,但你没有Newtonsoft的所有灵活性。但是,这里有一些在运行时排除属性的解决方案:
在 .NET 7 及更高版本中,可以控制序列化哪些属性,如下所述:https://devblogs.microsoft.com/dotnet/system-text-json-in-dotnet-7/#example-conditional-serialization
在 .NET 6(及更低版本)中 - 您可以强制转换为接口(我个人认为最干净)或使用映射器。这两个选项都在此答案中描述 https://stackoverflow.com/a/61344654/56621
[ScriptIgnore]
:
public class User
{
public int Id { get; set; }
public string Name { get; set; }
[ScriptIgnore]
public bool IsComplete
{
get { return Id > 0 && !string.IsNullOrEmpty(Name); }
}
}
在这种情况下,Id 和 name 将仅序列化
如果您不像我一样热衷于使用属性装饰代码,尤其是当您在编译时无法知道这里会发生什么时,这是我的解决方案。
使用 Javascript 序列化程序
public static class JsonSerializerExtensions
{
public static string ToJsonString(this object target,bool ignoreNulls = true)
{
var javaScriptSerializer = new JavaScriptSerializer();
if(ignoreNulls)
{
javaScriptSerializer.RegisterConverters(new[] { new PropertyExclusionConverter(target.GetType(), true) });
}
return javaScriptSerializer.Serialize(target);
}
public static string ToJsonString(this object target, Dictionary<Type, List<string>> ignore, bool ignoreNulls = true)
{
var javaScriptSerializer = new JavaScriptSerializer();
foreach (var key in ignore.Keys)
{
javaScriptSerializer.RegisterConverters(new[] { new PropertyExclusionConverter(key, ignore[key], ignoreNulls) });
}
return javaScriptSerializer.Serialize(target);
}
}
public class PropertyExclusionConverter : JavaScriptConverter
{
private readonly List<string> propertiesToIgnore;
private readonly Type type;
private readonly bool ignoreNulls;
public PropertyExclusionConverter(Type type, List<string> propertiesToIgnore, bool ignoreNulls)
{
this.ignoreNulls = ignoreNulls;
this.type = type;
this.propertiesToIgnore = propertiesToIgnore ?? new List<string>();
}
public PropertyExclusionConverter(Type type, bool ignoreNulls)
: this(type, null, ignoreNulls){}
public override IEnumerable<Type> SupportedTypes
{
get { return new ReadOnlyCollection<Type>(new List<Type>(new[] { this.type })); }
}
public override IDictionary<string, object> Serialize(object obj, JavaScriptSerializer serializer)
{
var result = new Dictionary<string, object>();
if (obj == null)
{
return result;
}
var properties = obj.GetType().GetProperties();
foreach (var propertyInfo in properties)
{
if (!this.propertiesToIgnore.Contains(propertyInfo.Name))
{
if(this.ignoreNulls && propertyInfo.GetValue(obj, null) == null)
{
continue;
}
result.Add(propertyInfo.Name, propertyInfo.GetValue(obj, null));
}
}
return result;
}
public override object Deserialize(IDictionary<string, object> dictionary, Type type, JavaScriptSerializer serializer)
{
throw new NotImplementedException(); //Converter is currently only used for ignoring properties on serialization
}
}
如果您使用的是System.Text.Json
那么您可以使用[JsonIgnore]
.
FQ: System.Text.Json.Serialization.JsonIgnoreAttribute
官方Microsoft文档:JsonIgnoreAttribute
如下所述:
该库作为 .NET Core 3.0 共享框架的一部分内置。
对于其他目标框架,请安装 System.Text.Json NuGet 包。该软件包支持:
- .NET 标准 2.0 及更高版本
- .NET Framework 4.6.1 及更高版本
- .NET Core 2.0、2.1 和 2.2
对于 C# 9 的记录,它是[property: JsonIgnore]
using System.Text.Json.Serialization;
public record R(
string Text2
[property: JsonIgnore] string Text2)
对于经典风格,它仍然只是[JsonIgnore]
。
using System.Text.Json.Serialization;
public record R
{
public string Text {get; init; }
[JsonIgnore]
public string Text2 { get; init; }
}
您也可以使用 [NonSerialized]
属性
[Serializable]
public struct MySerializableStruct
{
[NonSerialized]
public string hiddenField;
public string normalField;
}
从 MS 文档:
指示不应序列化可序列化类的字段。此类不能继承。
例如,如果您使用的是 Unity(这不仅适用于 Unity),那么这适用于UnityEngine.JsonUtility
using UnityEngine;
MySerializableStruct mss = new MySerializableStruct
{
hiddenField = "foo",
normalField = "bar"
};
Debug.Log(JsonUtility.ToJson(mss)); // result: {"normalField":"bar"}
Add System.Text.Json Version for dotnet core
对于编译时,按照上述答案中的建议添加 [JsonIgnore]。
对于运行时,需要将 JsonConverter 添加到选项中。
首先,为要排除的类型创建一个 JsonConverter,例如下面的ICollection<LabMethod>
public class LabMethodConverter : JsonConverter<ICollection<LabMethod>>
{
public override ICollection<LabMethod> Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
{
//deserialize JSON into a ICollection<LabMethod>
return null;
}
public override void Write(Utf8JsonWriter writer, ICollection<LabMethod> value, JsonSerializerOptions options)
{
//serialize a ICollection<LabMethod> object
writer.WriteNullValue();
}
}
然后在序列化 Json 时添加到选项
var options = new JsonSerializerOptions();
options.Converters.Add(new LabMethodConverter());
var new_obj = Json(new { rows = slice, total = count }, options);