当使用Newtonsoft.Json合并对象时,如何忽略空字符串值



我有一个数据模型,它被定义为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);

相关内容

  • 没有找到相关文章

最新更新