C# 在c中覆盖Json属性名#
我有一个包含以下字段的类。当需要调用外部RESTAPI方法时,这些属性用于序列化为json对象C# 在c中覆盖Json属性名#,c#,json,C#,Json,我有一个包含以下字段的类。当需要调用外部RESTAPI方法时,这些属性用于序列化为json对象 public class Customer { [JsonProperty(PropertyName = "email")] public string Email { get; set; } [JsonProperty(PropertyName = "prop[listId]")] public string Test{ ge
public class Customer
{
[JsonProperty(PropertyName = "email")]
public string Email { get; set; }
[JsonProperty(PropertyName = "prop[listId]")]
public string Test{ get; set; }
// there are lot of properties
}
在属性名Test
中,外部API服务调用需要类似以下json文件名格式的东西
prop[7]
在我的例子中,这个7
可以根据test、dev和prod等环境进行更改。因此,我正在寻找一种将listId值移动到app.config的方法
我试着按照下面的方法来做,但不允许这样做。对于listIdValue
如果分配常量值,它将起作用
private string listIdValue = ConfigurationManager.AppSettings["ListIdValue"];
[JsonProperty(PropertyName = "prop["+listIdValue +"]")]
public string Test{ get; set; }
定义不同的配置模式,如调试/发布/质量保证/登台 然后为它们中的每一个添加编译符号。在代码中,您可以执行以下操作: 我想您定义了以下内容:QA和STAGING
public class Customer
{
[JsonProperty(PropertyName = "email")]
public string Email { get; set; }
#if QA
[JsonProperty(PropertyName = "prop[QA_ID]")]
#elif STAGING
[JsonProperty(PropertyName = "prop[STAGING_ID]")]
#endif
public string Test{ get; set; }
// there are lot of properties
}
您也可以使用这些配置进行自动部署,这将节省您的时间。您必须覆盖
DefaultContractResolver
,并实现您自己的机制来提供PropertyName
(JSON格式)。我将提供一个完整的示例代码,用运行时生成的PropertyName
显示反序列化和序列化。目前,它将Test
字段修改为Test5
(在所有型号中)。您应该实现自己的机制(使用属性、保留名称、表或任何东西)
class Program
{
static void Main(string[] args)
{
var customer = new Customer() {Email = "asd@asd.com", Test = "asdasd"};
var a = Serialize(customer, false);
var b = Serialize(customer, true);
Console.WriteLine(a);
Console.WriteLine(b);
var desA = Deserialize<Customer>(a, false);
var desB = Deserialize<Customer>(b, true);
Console.WriteLine("TestA: {0}", desA.Test);
Console.WriteLine("TestB: {0}", desB.Test);
}
static string Serialize(object obj, bool newNames)
{
JsonSerializerSettings settings = new JsonSerializerSettings();
settings.Formatting = Formatting.Indented;
if (newNames)
{
settings.ContractResolver = new CustomNamesContractResolver();
}
return JsonConvert.SerializeObject(obj, settings);
}
static T Deserialize<T>(string text, bool newNames)
{
JsonSerializerSettings settings = new JsonSerializerSettings();
settings.Formatting = Formatting.Indented;
if (newNames)
{
settings.ContractResolver = new CustomNamesContractResolver();
}
return JsonConvert.DeserializeObject<T>(text, settings);
}
}
class CustomNamesContractResolver : DefaultContractResolver
{
protected override IList<JsonProperty> CreateProperties(System.Type type, MemberSerialization memberSerialization)
{
// Let the base class create all the JsonProperties
// using the short names
IList<JsonProperty> list = base.CreateProperties(type, memberSerialization);
// Now inspect each property and replace the
// short name with the real property name
foreach (JsonProperty prop in list)
{
if (prop.UnderlyingName == "Test") //change this to your implementation!
prop.PropertyName = "Test" + 5;
}
return list;
}
}
public class Customer
{
[JsonProperty(PropertyName = "email")]
public string Email { get; set; }
public string Test { get; set; }
}
如您所见,当我们使用Serialize(…,false)
时,字段的名称是Test
,当我们使用Serialize(…,true)
时,字段的名称是Test5
,这与预期的一样。这也适用于反序列化
我已经使用这个答案作为我的答案的操作:为什么?为什么需要这些json属性?您有很多属性吗?如果序列化库(json.Net?)失败,您可能想“手动”序列化,只需在
JObject
上设置属性使用TypeDescriptor.GetAttributes
进行反射,应该可以在运行时添加这些属性。您使用的是哪个库?我已经修改了问题以添加详细信息。在对JSON进行反序列化之前,请查找并替换使用任何类型调试属性名称的JSON文本。OnBeforeDeserialization+=(s,t)=>t.替换(“prop[x],“prop[y]”
但即使这个列表id也可以随时更改,如果我在类级别硬编码这些列表值,那么代码级别也会更改。
{
"email": "asd@asd.com",
"Test": "asdasd"
}
{
"email": "asd@asd.com",
"Test5": "asdasd"
}
TestA: asdasd
TestB: asdasd