将包含多个json对象的字符串解析为更方便的东西

时间:2016-07-14 16:09:31

标签: c# c#-3.0

问题的关键在于我不知道任何C#,但发现自己在某些测试基础设施中添加了一个功能,恰好用C#编写。我怀疑这个问题完全是微不足道的,请耐心等待回答。我最初写这些东西的同事都不在办公室。

我正在解析表示一个或多个json对象的字符串。到目前为止,我可以得到第一个对象,但无法弄清楚如何访问其余部分。

public class demo
{
public void minimal()
{
    // Note - the input is not quite json! I.e. I don't have 
    // [{"Name" : "foo"}, {"Name" : "bar"}]
    // Each individual object is well formed, they just aren't in
    // a convenient array for easy parsing.
    // Each string representation of an object are literally concatenated.

    string data = @"{""Name"": ""foo""} {""Name"" : ""bar""}";

    System.Xml.XmlDictionaryReader jsonReader = 
       JsonReaderWriterFactory.CreateJsonReader(Encoding.UTF8.GetBytes(data),
       new System.Xml.XmlDictionaryReaderQuotas());

    System.Xml.Linq.XElement root = XElement.Load(jsonReader);
    Assert.AreEqual(root.XPathSelectElement("//Name").Value, "foo");

    // The following clearly doesn't work
    Assert.AreEqual(root.XPathSelectElement("//Name").Value, "bar");
}
}

我大致已经处理了足够的解析器来计算通过计算括号来分割字符串的位置,但我希望库支持会为我做这个。

理想的最终结果是您选择的顺序数据结构(列表,向量?不关心),其中包含嵌入在字符串中的每个json对象的一个​​System.Xml.Linq.XElement

谢谢!

编辑:大致可行的例子,主要是由于George Richardson - 我在类型系统中玩得很快(不确定C#3.0中是否有动态),但最终结果似乎是可预测的。

public class demo
{
    private IEnumerable<Newtonsoft.Json.Linq.JObject>
            DeserializeObjects(string input)
    {
        var serializer = new JsonSerializer();
        using (var strreader = new StringReader(input))
        {
            using (var jsonreader = new JsonTextReader(strreader))
            {
                jsonreader.SupportMultipleContent = true;
                while (jsonreader.Read())
                {
                    yield return (Newtonsoft.Json.Linq.JObject)
                                  serializer.Deserialize(jsonreader);
                }
            }
        }
    }

    public void example()
    {
        string json = @"{""Name"": ""foo""} {""Name"" : ""bar""} {""Name"" : ""baz""}";
        var objects = DeserializeObjects(json);

        var array = objects.ToArray();
        Assert.AreEqual(3, array.Length);
        Assert.AreEqual(array[0]["Name"].ToString(), "foo");
        Assert.AreEqual(array[1]["Name"].ToString(), "bar");
        Assert.AreEqual(array[2]["Name"].ToString(), "baz");
    }
}

1 个答案:

答案 0 :(得分:2)

您将要使用JSON.net来满足实际的反序列化需求。我在这里看到的一个大问题是你的json数据正在被连接在一起,这意味着你将不得不从字符串中提取每个对象。幸运的是,json.net的JsonReader有一个SupportMultipleContent属性,只有这个

public void Main()
{
    string json = @"{""Name"": ""foo""} {""Name"" : ""bar""} {""Name"" : ""baz""}";
    IEnumerable<dynamic> deserialized = DeserializeObjects(json);
    string name = deserialized.First().Name; //name is "foo"
}

IEnumerable<object> DeserializeObjects(string input)
{
    JsonSerializer serializer = new JsonSerializer();
    using (var strreader = new StringReader(input)) {
        using (var jsonreader = new JsonTextReader(strreader)) {
            jsonreader.SupportMultipleContent = true;
            while (jsonreader.Read()) {
                yield return serializer.Deserialize(jsonreader);
            }
        }
    }
}