【问题标题】:JSON parse error in iOSiOS中的JSON解析错误
【发布时间】:2015-02-22 00:18:11
【问题描述】:

我正在尝试解析 json 响应(我从 SOAP Web 服务响应的结果标记中获取),如下图所示。

NSString *soapMessage = [NSString stringWithFormat:@"<?xml version=\"1.0\" encoding=\"utf-8\"?>"
                         "<soap:Envelope xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\">"
                         "<soap:Body>"
                         "<GetPhotoSession xmlns=\"http://tempuri.org/\">"
                         "<UserID>%@</UserID>"
                         "<photoSessionID>%@</photoSessionID>"
                         "</GetPhotoSession>"
                         "</soap:Body>"
                         "</soap:Envelope>",[HelperClass retrieveStringForKey:kUserID],self.currentSession.sessionID];

NSURL *url = [NSURL URLWithString:kBaseURL];
NSMutableURLRequest *theRequest = [NSMutableURLRequest requestWithURL:url];
NSString *msgLength = [NSString stringWithFormat:@"%lu", (unsigned long)[soapMessage length]];

[theRequest addValue: @"text/xml; charset=utf-8" forHTTPHeaderField:@"Content-Type"];
[theRequest addValue: @"http://tempuri.org/GetPhotoSession" forHTTPHeaderField:@"SOAPAction"];
[theRequest addValue: msgLength forHTTPHeaderField:@"Content-Length"];
[theRequest setHTTPMethod:@"POST"];
[theRequest setHTTPBody: [soapMessage dataUsingEncoding:NSUTF8StringEncoding]];

NSURLConnection *theConnection = [[NSURLConnection alloc] initWithRequest:theRequest delegate:self];

if( theConnection ) {
    self.webResponseData = [NSMutableData data];
}else {
    NSLog(@"Some error occurred in Connection");

}

-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
    [self.webResponseData  setLength:0];
}

-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
    [self.webResponseData  appendData:data];
}

-(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
    NSLog(@"Some error in your Connection. Please try again.");
}

-(void)connectionDidFinishLoading:(NSURLConnection *)connection {
    NSLog(@"Received Bytes from server: %d", [self.webResponseData length]);
    NSString *myXMLResponse = [[NSString alloc] initWithBytes: [self.webResponseData bytes] length:[self.webResponseData length] encoding:NSUTF8StringEncoding];
    NSLog(@"%@",myXMLResponse);

    NSError *errorPointer;

    NSDictionary *dict = [XMLReader dictionaryForXMLString:myXMLResponse error:&errorPointer];

    NSString *jsonData = dict[@"soap:Envelope"][@"soap:Body"][@"GetPhotoSessionResponse"][@"GetPhotoSessionResult"][@"text"];

    NSData *data = [jsonData dataUsingEncoding:NSUTF8StringEncoding];
    id json = [NSJSONSerialization JSONObjectWithData:data options:0 error:&errorPointer];

    NSLog(@"%@",[json objectForKey:@"Head"]);
}

但我在“json”对象中得到了 nil。这是 JSON 响应 http://pastie.org/9799331 的粘贴链接。以下是错误指针的说明。

errorPointer的打印说明: 错误域 = NSCocoaErrorDomain 代码 = 3840“操作无法完成。(可可错误 3840。)”(JSON 文本未以数组或对象开头,并且允许未设置片段的选项。) UserInfo = 0x7d0156a0 {NSDebugDescription = JSON 文本没有以数组或对象开头,并且没有设置允许片段的选项。}

【问题讨论】:

  • 将您的 JSON 粘贴到 JSONLint 并检查它是否有效
  • JSONLint 的结果:第 5 行解析错误:... "ImageBase64Data": "/9j/4AAQSkZJRgABAQA --------- --^ 期望 'STRING'、'NUMBER'、'NULL'、'TRUE'、'FALSE'、'{'、'['
  • ImageBase64Data 键值无效,请检查,值之间是否有换行符或"
  • @Rob No. 这是 标签的值,它是一个 JSON 响应
  • 好的。我之所以追求这一点,是因为您报告的错误消息(“JSON 文本不是以数组或对象开头 ...”)与传递给 NSJSONSerialization 的非 JSON 更一致,而不是 Midhun/Zaph 所追求的,即它是 JSON,但格式不正确。通常格式错误的 JSON 会给出更具描述性的错误。您的错误表明了一个更根本的问题。我可能会建议NSLogdata(应该以十六进制显示),这样我们就可以确认没有奇怪的BOM 或类似的东西。只需向我们展示它的开始......

标签: ios json web-services soap


【解决方案1】:

这是您自己的 Web 服务还是现有的?如果这是您自己的,您确定您的 JSON 实际上是正确的吗?

我相信问题出在这里:

"ImageBase64Data": "/9j/4AAQSkZJRgABAQA .... blah blah ==
"

注意这里的关键点。结束双引号在下一行。应该是

"ImageBase64Data": "/9j/4AAQSkZJRgABAQA .... blah blah =="

为了将来参考,如果您在问题中剪切和粘贴 JSON 有效负载文本,比显示图像要容易得多。

所以从这里获取您的数据:

http://pastie.org/9799331#2,22,482

我已经通过这段代码运行了它:

NSError *error = nil;
NSString *path = [[NSBundle mainBundle] pathForResource:@"badjson" ofType:@"txt"];
NSString *jsonStr = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:nil];

//    jsonStr = [[jsonStr componentsSeparatedByCharactersInSet:[NSCharacterSet newlineCharacterSet]] componentsJoinedByString:@""];

NSData *jsonData = [jsonStr dataUsingEncoding:NSUTF8StringEncoding];

id json = [NSJSONSerialization JSONObjectWithData:jsonData options:kNilOptions error:&error];

NSLog(@"Error is %@", error);
NSLog(@"JSON is %@", json);

请注意,我现在有一个注释掉的行。当我这样做时,它失败了。但是,失败与您上面列出的不同。我明白了。

Error is Error Domain=NSCocoaErrorDomain Code=3840 "操作 无法完成。 (可可错误 3840。)”(未转义的控制 字符 120 周围的字符。) UserInfo=0x7f9519f71510 {NSDebugDescription=字符 120 周围未转义的控制字符。}

如果我取消注释该行,那么它可以工作。只显示输出的一个sn-p:

2014-12-26 12:03:35.020 Sample[49732:6379864] Error is (null)
2014-12-26 12:03:35.022 Sample[49732:6379864] JSON is {
     Head =     (
                 {
             ID = 1092;
             ImageBase64Data = "/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDACAWGBwYFCAcGhwkIiAmMFA0MCwsMGJGSjpQdGZ6eHJmcG6AkLicgIiuim5woNqirr7EztDOfJri8uDI8LjKzsb/

您的数据使用 base64 输出,该输出使用换行符分隔每个编码行。这就是为什么您会在末尾看到如下文本:

==
","ImageName"

您还会注意到 Base64 中的绝大多数行看起来是统一的 .. 由于这种分隔。

使用这个去掉换行符(你需要为你的代码修改它)

jsonStr = [[jsonStr componentsSeparatedByCharactersInSet:[NSCharacterSet newlineCharacterSet]] componentsJoinedByString:@""];

如果这不起作用,那么您需要提供更多信息:

-输出是从 Xcode 控制台获取的吗? - 在您的代码中添加 NSLog 以获取输出(如果您没有,请添加一个)。 - 添加 NSLog 后,提供该输出

虽然看起来您“已经提供了该信息”,但信息显然存在差距。否则我希望看到相同的 NSError 输出。所以我们要么缺少步骤,要么可能存在其他差异。例如,我在 iOS 8 模拟器中运行它。并不是说它应该太重要,但总有可能您运行在以不同方式解析 JSON 的旧版 iOS 上。

【讨论】:

  • 我同意 JSON 无效。而且它不仅仅是 base-64 字符串末尾的新行,而是每隔 72 或 80 个字符贯穿整个字符串。换行符绝对不应该出现在 JSON 字符串中(正如 Midhun MP 指出的那样)。但是我想知道是否没有更深层次的问题,因为NSJSONSerialization 错误不是在谈论未转义的控制字符,而是根本没有找到 JSON。
  • @Rob,你是对的。该错误会提到未转义的控制字符。我也认为 ChampTaurus 应该显示 NSData 转储是对的。尽管 BOM 会在控制台中显示为字符(我刚刚对此进行了测试)。不过,它可能是另一个不可打印的字符。
  • @MobileBen 我不这么认为。请使用粘贴链接检查我更新的问题。
  • @ChampTaurus,输出粘贴链接中的数据的代码是什么?我假设这是从 Xcode 控制台直接剪切和粘贴的?你能更新你的代码并显示你在NSLog'ing JSON 字符串的位置吗?这里的问题是 Base 64 实际上有大量的换行符。看起来正在输出的 Base64 被换行符分隔。请参阅我更新的答案,其中详细说明了一点。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-10-27
  • 2012-06-25
  • 2013-07-16
  • 1970-01-01
  • 2010-11-22
  • 1970-01-01
相关资源
最近更新 更多