ServiceStack.Text json反序列化会创建错误的对象,而不是抛出无效的json输入字符串

时间:2012-04-14 12:11:23

标签: c# json servicestack

当我尝试反序列化这个无效的json字符串时(}]最后丢失):

[{"ExtId":"2","Name":"VIP sj�lland","Mobiles":["4533333333","4544444444"]

通过这样做:

var result = JsonSerializer.DeserializeFromString<T>(str);    

ServiceStack json反序列化器接受字符串,但它会创建一个错误的对象,因为我最终得到了一个具有以下值的C#对象:

ExtId : "2"                                      // ok fine.
Name: "VIP sj�lland"                            // ok fine
Mobiles: ["4533333333","4544444444", "544444444"]// Aarg! An array with 3 objects ?!? 
                                                 // There were only two in the JSON string.

在这种情况下,抛出异常而不是继续使用错误数据会好得多。因此我尝试使用:

JsConfig.ThrowOnDeserializationError = true; 

在调用DeserializeFromString之前,但没有抛出任何异常。在1月份,我问了这个问题Configure ServiceStack.Text to throw on invalid JSON,答案是ServiceStack支持静音,我可以在GitHub中发出拉取请求。

这仍然是这样吗?有没有人已经做过,省了我的麻烦?否则,我的时间表非常紧张,所以如果有人有一些代码或建议如何创建一个选项标志以使ServiceStack抛出反序列化错误,请回复此处,以便我可以更快地完成此操作。

4 个答案:

答案 0 :(得分:1)

这在ServiceStack.Text v4 + 中得到解决,默认情况下不会填充不完整的集合,例如:

public class Poco
{
    public string ExtId { get; set; }
    public string Name { get; set; }
    public string[] Mobiles { get; set; }
}

var json = "[{\"ExtId\":\"2\",\"Name\":\"VIP sj�lland\",\"Mobiles\":[\"4533333333\",\"4544444444\"]";

var dto = json.FromJson<Poco[]>();

Assert.That(dto[0].ExtId, Is.EqualTo("2"));
Assert.That(dto[0].Name, Is.EqualTo("VIP sj�lland"));
Assert.That(dto[0].Mobiles, Is.Null);

或者如果首选可以抛出错误:

JsConfig.ThrowOnDeserializationError = true;

Assert.Throws<SerializationException>(() => 
    json.FromJson<Poco[]>());

答案 1 :(得分:0)

对于JSON来说,C#有点挑剔。以下将有效!注意我没有匿名对象数组作为默认元素。

{
    "ExtItem": [
        {
            "ExtId": "2",
            "Name": "VIPsj�lland",
            "Mobiles": [
                "4533333333",
                "4544444444"
            ]
        }
    ]
}

如果我从中生成POCO,我会

public class Rootobject
{
    public Extitem[] ExtItem { get; set; }
}

public class Extitem
{
    public string ExtId { get; set; }
    public string Name { get; set; }
    public string[] Mobiles { get; set; }
}

我个人使用扩展方法来字符串

public static class Extensions
{
    public  static bool DeserializeJson<T>(this String str, out T item)
    {
        item = default(T);
        try
        {
            item = new JavaScriptSerializer().Deserialize<T>(str);
            return true;
        }
        catch (Exception ex)
        {
            return false;
        }
    }
}

这将使我能够写下:

Rootobject ext;
const string validJson = @"
{
    ""ExtItem"": [
        {
            ""ExtId"":""2"",
            ""Name"":""VIPsj�lland"",
            ""Mobiles"":[
                ""4533333333"",
                ""4544444444""
            ]
        }
    ]
}";
if (validJson.DeserializeJson(out ext))
{ //valid input 
    // following would print 2 elements : 4533333333, 4544444444
    Console.WriteLine(string.Join(", ", ext.ExtItem.First().Mobiles)); 
} //invalid input

答案 2 :(得分:0)

我试过这个,当我在最后错过了}时抛出异常。

在C#中,JSON的格式是{&#34; name&#34;,&#34; value&#34;}而不是[{&#34; name&#34;,&#34; value&#34; }]。

class M
        {
            public string ExtId { get; set; }
            public string Name { get; set; }
            public List<string> Mobiles { get; set; }
        }

string str = "{\"ExtId\":\"2\",\"Name\":\"VIP\",\"Mobiles\":[\"4533333333\",\"4544444444\"]";

M m = JsonConvert.DeserializeObject<M>(str);

运行此操作时,如果缺少},则会收到错误。

使用:

string str = "{\"ExtId\":\"2\",\"Name\":\"VIP\",\"Mobiles\":[\"4533333333\",\"4544444444\"]}";

该对象被反序列化。

您可以从以下位置查看此对象的JSON:

string s = JsonConvert.SerializeObject(m);

答案 3 :(得分:-1)

我只使用Newtonsoft.Json T JsonConvert.DeserializeObject<T>(string value)并抛出异常;

如果我使用object JsonConvert.DeserializeObject(string value)这个,只需放置丢失的}]

即可创建正确的对象

我发现这是一个非常可靠和快速的库。