我在DLL中有一个类,它没有标有DataContract,JsonProperty
等。现在,我想将该类的实例序列化为 JSON 对象,并缩短 C# 属性名称。
例如,该类是:
public class Foo
{
public string SomeLengthyCSharpPropertyName { get; set; }
}
我想知道我是否可以在 C# 名称和 json 名称之间创建映射。我不能直接添加数据契约,Json属性属性
,如下所示。有什么解决方法吗?
[DataContract]
public class Foo
{
[JsonProperty("s")]
public string SomeLengthyCSharpPropertyName { get; set; }
}
我倾向于不创建另一个具有相同但 JsonProperty
装饰属性的类,并将这些属性复制到新类,然后序列化。
您可以使用成员的覆盖属性字典创建自己的自定义 ContractResolver
,然后重写 CreateProperty()
并将覆盖应用于基类返回的 JsonProperty
:
public class JsonPropertyOverride
{
public string PropertyName { get; set; }
public bool? Ignored { get; set; }
// Others as required from http://www.newtonsoft.com/json/help/html/T_Newtonsoft_Json_JsonPropertyAttribute.htm
// Changing all value type properties to nullables.
}
public class OverrideContractResolver : DefaultContractResolver
{
readonly Dictionary<MemberInfo, JsonPropertyOverride> overrides; // A private copy for thread safety.
public OverrideContractResolver(IDictionary<MemberInfo, JsonPropertyOverride> overrides)
: base()
{
if (overrides == null)
throw new ArgumentNullException();
this.overrides = overrides.ToDictionary(p => p.Key, p => p.Value);
}
protected override JsonProperty CreateProperty(MemberInfo member, MemberSerialization memberSerialization)
{
var property = base.CreateProperty(member, memberSerialization);
if (property != null)
{
JsonPropertyOverride attr;
if (overrides.TryGetValue(member, out attr))
{
if (attr.PropertyName != null)
property.PropertyName = ResolvePropertyName(attr.PropertyName);
if (attr.Ignored != null)
property.Ignored = attr.Ignored.Value;
}
}
return property;
}
}
如果您愿意,也可以从 CamelCasePropertyNamesContractResolver
继承。
然后像这样使用它:
public class Foo
{
public string SomeLengthyCSharpPropertyName { get; set; }
public string DefaultNotIgnored { get; set; }
[JsonIgnore]
public string DefaultIgnored { get; set; }
}
public class TestClass
{
public static void Test()
{
var foo = new Foo { SomeLengthyCSharpPropertyName = "SomeLengthyCSharpPropertyName", DefaultIgnored = "DefaultIgnored", DefaultNotIgnored = "DefaultNotIgnored" };
var resolver = new OverrideContractResolver(new Dictionary<MemberInfo, JsonPropertyOverride> {
{ typeof(Foo).GetProperty("SomeLengthyCSharpPropertyName"), new JsonPropertyOverride { PropertyName = "c" } },
{ typeof(Foo).GetProperty("DefaultNotIgnored"), new JsonPropertyOverride { Ignored = true } },
{ typeof(Foo).GetProperty("DefaultIgnored"), new JsonPropertyOverride { Ignored = false } },
});
var settings = new JsonSerializerSettings { ContractResolver = resolver };
var json = JsonConvert.SerializeObject(foo, settings); // Outputs {"c":"SomeLengthyCSharpPropertyName","DefaultIgnored":"DefaultIgnored"}
Debug.WriteLine(json);
var expectedJson = @"{ ""c"": ""SomeLengthyCSharpPropertyName"", ""DefaultIgnored"": ""DefaultIgnored"" }";
var ok = JToken.DeepEquals(JToken.Parse(json), JToken.Parse(expectedJson));
Debug.Assert(ok); // No assert
var foo2 = JsonConvert.DeserializeObject<Foo>(json, settings);
var ok2 = foo2.DefaultIgnored == foo.DefaultIgnored && foo2.SomeLengthyCSharpPropertyName == foo.SomeLengthyCSharpPropertyName;
Debug.Assert(ok2); // No assert
}
}
这绝对感觉像是一种解决方法,但您可能需要考虑一下。如果类未密封,则可以从它继承,重写它的属性(要更改的属性),然后修饰它们。