我有一个数据模型,它被定义为C#中的一个类。我需要使用JObject.merge合并两个对象,但在一个特定属性的情况下,我需要忽略第二个对象中的空字符串值,类似于忽略null值的方式。这是否有一个现有的属性,或者我是否需要编写自己的属性?
void Main()
{
string json1 = @"{ Foo: ""foo1"", Bar: ""bar1"" }";
string json2 = @"{ Foo: ""foo2"", Bar: null }";
string json3 = @"{ Foo: ""foo3"", Bar: """" }";
var model1 = JObject.Parse(json1);
var model2 = JObject.Parse(json2);
model1.Merge(model2);
model1.Dump();
model1 = JObject.Parse(json1);
model2 = JObject.Parse(json3);
model1.Merge(model2);
model1.Dump();
}
public class Model
{
[JsonProperty("foo")]
public string Foo { get ; set; }
[JsonProperty("bar", NullValueHandling = NullValueHandling.Ignore)]
public string Bar { get ; set; }
}
Output (1): Model.Bar = "bar1"
Output (2): Model.Bar = "";
Desired output of (2): Model.Bar = "bar1"
EDIT:好的,我意识到不能应用属性,因为我只需要将原始json字符串作为输入。这主要是因为我的类具有默认值的属性。合并具有空值的类会触发默认值,并最终覆盖原始值,这是我不希望的。我需要能够获取类的部分json表示,并更新原始表示。如果从一开始就不清楚的话,我很抱歉。我会更新我的答案。
您可以使用以下扩展方法从要合并的JObject
中删除具有空字符串值的属性:
public static class JsonExtensions
{
public static void RemovePropertiesByValue(this JToken root, Predicate<JValue> filter)
{
var nulls = root.DescendantsAndSelf().OfType<JValue>().Where(v => v.Parent is JProperty && filter(v)).ToList();
foreach (var value in nulls)
{
var parent = (JProperty)value.Parent;
parent.Remove();
}
}
public static IEnumerable<JToken> DescendantsAndSelf(this JToken node)
{
if (node == null)
return Enumerable.Empty<JToken>();
var container = node as JContainer;
if (container != null)
return container.DescendantsAndSelf();
else
return new [] { node };
}
}
然后像这样使用:
model2.RemovePropertiesByValue(v => v.Type == JTokenType.String && string.IsNullOrEmpty((string)v.Value));
请注意,这不会从数组中删除空字符串,因为这会扰乱数组索引,从而导致合并。你也需要吗?
我操作JObject字典键来处理特定条目。我觉得很脏,但它有效。我想不出一个"好"的方法来做这件事。
model1 = JObject.Parse(json1);
model2 = JObject.Parse(json3);
IDictionary<string, JToken> dictionary = model2;
dictionary["Bar"] = string.IsNullOrEmpty((string)dictionary["Bar"])
? null
: dictionary["Bar"];
model1.Merge(model2);