วิธีใช้ NSJSONSerialization


156

ฉันมีสตริง JSON (จาก PHP json_encode()ที่มีลักษณะเช่นนี้:

[{"id": "1", "name":"Aaa"}, {"id": "2", "name":"Bbb"}]

ฉันต้องการแยกวิเคราะห์สิ่งนี้ลงในโครงสร้างข้อมูลบางอย่างสำหรับแอพ iPhone ของฉัน ผมคิดว่าสิ่งที่ดีที่สุดสำหรับผมที่จะมีอาร์เรย์ของพจนานุกรมดังนั้นองค์ประกอบ 0 ในอาร์เรย์เป็นพจนานุกรมที่มีปุ่มและ"id" => "1""name" => "Aaa"

ฉันไม่เข้าใจวิธีการจัดNSJSONSerializationเก็บข้อมูลแม้ว่า นี่คือรหัสของฉัน:

NSError *e = nil;
NSDictionary *JSON = [NSJSONSerialization 
    JSONObjectWithData: data 
    options: NSJSONReadingMutableContainers 
    error: &e];

นี่เป็นเพียงสิ่งที่ฉันเห็นเป็นตัวอย่างในเว็บไซต์อื่น ผมได้พยายามที่จะได้รับการอ่านในที่JSONวัตถุโดยการพิมพ์ออกมาจำนวนขององค์ประกอบและสิ่งที่ต้องการ EXC_BAD_ACCESSแต่ฉันมักจะได้รับ

ฉันจะใช้NSJSONSerializationวิเคราะห์ JSON ด้านบนและแปลงเป็นโครงสร้างข้อมูลที่ฉันกล่าวถึงได้อย่างไร


ตัวแปรข้อมูลของคุณน่าจะเป็นศูนย์
d.lebedev

ไม่ใช่ฉันได้ทำการทดสอบแล้ว
Logan Serman

คุณได้ลองดูว่ามีข้อมูลที่เกี่ยวข้องในวัตถุข้อผิดพลาดหรือไม่?
Monolo

คำตอบ:


214

วัตถุ json รูทของคุณไม่ใช่พจนานุกรม แต่เป็นอาร์เรย์:

[{"id": "1", "name":"Aaa"}, {"id": "2", "name":"Bbb"}]

นี่อาจให้ภาพที่ชัดเจนเกี่ยวกับวิธีจัดการกับมัน:

NSError *e = nil;
NSArray *jsonArray = [NSJSONSerialization JSONObjectWithData: data options: NSJSONReadingMutableContainers error: &e];

if (!jsonArray) {
  NSLog(@"Error parsing JSON: %@", e);
} else {
   for(NSDictionary *item in jsonArray) {
      NSLog(@"Item: %@", item);
   }
}

ขอบคุณฉันจะลอง แต่ไม่ควร[JSON count]ส่งคืนสิ่งใดแทนที่จะมอบ EXC_BAD_ACCESS ให้ฉัน
Logan Serman

ควรว่าทำไมฉันเพิ่มการตรวจสอบถ้า!jsonArrayและพิมพ์ข้อผิดพลาด สิ่งนี้ควรแสดงข้อผิดพลาดที่เกิดขึ้นระหว่างการวิเคราะห์คำ
rckoenes

1
@ xs2bush ไม่ได้เนื่องจากคุณไม่ได้สร้างjsonArrayมันควรจะปิดอัตโนมัติ
rckoenes

@Logan: ใช่ [จำนวน JSON] ควรส่งคืนค่า ดูคำตอบของฉันด้านล่างเกี่ยวกับซอมบี้ EXC_BAD_ACCESS เกือบจะเกี่ยวข้องกับซอมบี้เสมอ
Olie

ในกรณีนี้รายการเป็นกุญแจสำคัญในคู่ค่าคีย์ JSON ที่กำหนด .. สำหรับลูปของคุณทำงานได้อย่างสมบูรณ์แบบการส่งออกแต่ละคีย์ JSON ของฉัน ฉันรู้แล้วว่ากุญแจสำหรับค่าที่ฉันต้องการคือ 'กุญแจ' ความพยายามในการรับค่าของคีย์นี้และส่งออกไปยังบันทึกล้มเหลว มีความเข้าใจเพิ่มเติมหรือไม่
Thomas Clowes

75

นี่คือรหัสของฉันสำหรับตรวจสอบว่า json ที่ได้รับเป็นอาร์เรย์หรือพจนานุกรม:

NSError *jsonError = nil;
id jsonObject = [NSJSONSerialization JSONObjectWithData:jsonData options:kNilOptions error:&jsonError];

if ([jsonObject isKindOfClass:[NSArray class]]) {
    NSLog(@"its an array!");
    NSArray *jsonArray = (NSArray *)jsonObject;
    NSLog(@"jsonArray - %@",jsonArray);
}
else {
    NSLog(@"its probably a dictionary");
    NSDictionary *jsonDictionary = (NSDictionary *)jsonObject;
    NSLog(@"jsonDictionary - %@",jsonDictionary);
}

ฉันลองตัวเลือกนี้แล้ว: kNilOptions และ NSJSONReadingMutableContainers และทำงานอย่างถูกต้องสำหรับทั้งสอง

เห็นได้ชัดว่ารหัสจริงไม่สามารถเป็นแบบนี้ได้ซึ่งฉันสร้างตัวชี้ NSArray หรือ NSDictionary ภายในบล็อก if-else


29

มันใช้งานได้สำหรับฉัน dataวัตถุของคุณอาจเป็นไปได้nilและตามที่ rckoenes บันทึกไว้วัตถุรากควรเป็นอาร์เรย์ ดูรหัสนี้:

NSString *jsonString = @"[{\"id\": \"1\", \"name\":\"Aaa\"}, {\"id\": \"2\", \"name\":\"Bbb\"}]";
NSData *jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding];
NSError *e = nil;
NSMutableArray *json = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingMutableContainers error:&e];
NSLog(@"%@", json);

(ฉันต้องหลีกเลี่ยงอัญประกาศในสตริง JSON ด้วยแบ็กสแลช)


9

รหัสของคุณดูเหมือนดียกเว้นผลลัพธ์คือNSArrayไม่ใช่ไม่ใช่NSDictionaryนี่คือตัวอย่าง:

สองบรรทัดแรกเพียงสร้างวัตถุข้อมูลด้วย JSON เช่นเดียวกับที่คุณจะได้อ่านจากสุทธิ

NSString *jsonString = @"[{\"id\": \"1\", \"name\":\"Aaa\"}, {\"id\": \"2\", \"name\":\"Bbb\"}]";
NSData *jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding];

NSError *e;
NSMutableArray *jsonList = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingMutableContainers error:&e];
NSLog(@"jsonList: %@", jsonList);

เนื้อหา NSLog (รายการพจนานุกรม):

jsonList: (
           {
               id = 1;
               name = Aaa;
           },
           {
               id = 2;
               name = Bbb;
           }
           )

ตัวเลือกนี้ (NSJSONReadingMutableContainers) หมายถึงอะไร ฉันไม่ kNilOption และทุกอย่างทำงานได้ดี บอกวัตถุประสงค์ของการใช้ตัวเลือกเหล่านี้
Zar E Ahmer

Hit สูงสุดใน Google:: NSJSONReadingMutableLeaves"ระบุว่า leaf leaf ในกราฟวัตถุ JSON ถูกสร้างเป็นอินสแตนซ์ของ NSMutableString"
zaph

และสิ่งที่เกี่ยวกับ MutableContainer
Zar E Ahmer

โอ๊ะอีกครั้งจากผลลัพธ์ของ Google อันดับต้น ๆNSJSONReadingMutableContainers:: "ระบุว่าอาร์เรย์และพจนานุกรมถูกสร้างเป็นวัตถุที่ไม่แน่นอน"
zaph

1
สิ่งเหล่านี้จะช่วยหากคุณวางแผนที่จะแก้ไขออบเจ็กต์ JSON ที่ส่งคืนและบันทึกไว้ ไม่ว่าในกรณีใดวัตถุอาจเป็นวัตถุที่ถูกลบอัตโนมัติและดูเหมือนว่าจะเป็นสาเหตุของปัญหา
Deepak GM

6
[{"id": "1", "name":"Aaa"}, {"id": "2", "name":"Bbb"}]

ในข้อมูล JSON ด้านบนคุณกำลังแสดงว่าเรามีอาร์เรย์ที่มีจำนวนพจนานุกรม

คุณต้องใช้รหัสนี้ในการแยกวิเคราะห์:

NSError *e = nil;
NSArray *JSONarray = [NSJSONSerialization JSONObjectWithData: data options: NSJSONReadingMutableContainers error: &e];
        for(int i=0;i<[JSONarray count];i++)
        {
            NSLog(@"%@",[[JSONarray objectAtIndex:i]objectForKey:@"id"]);
             NSLog(@"%@",[[JSONarray objectAtIndex:i]objectForKey:@"name"]);
        }

สำหรับรวดเร็ว 3/3 +

   //Pass The response data & get the Array
    let jsonData = try JSONSerialization.jsonObject(with: data!, options: .allowFragments) as! [AnyObject]
    print(jsonData)
    // considering we are going to get array of dictionary from url

    for  item  in jsonData {
        let dictInfo = item as! [String:AnyObject]
        print(dictInfo["id"])
        print(dictInfo["name"])
    }

3

รหัสต่อไปนี้ดึงวัตถุ JSON จากเว็บเซิร์ฟเวอร์และแยกวิเคราะห์มันเป็น NSDictionary ฉันใช้ openweathermap API ที่ส่งกลับการตอบสนอง JSON อย่างง่ายสำหรับตัวอย่างนี้ เพื่อให้ง่ายรหัสนี้ใช้การร้องขอแบบซิงโครนัส

   NSString *urlString   = @"http://api.openweathermap.org/data/2.5/weather?q=London,uk"; // The Openweathermap JSON responder
   NSURL *url            = [[NSURL alloc]initWithString:urlString];
   NSURLRequest *request = [NSURLRequest requestWithURL:url];
   NSURLResponse *response;
   NSData *GETReply      = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:nil];
   NSDictionary *res     = [NSJSONSerialization JSONObjectWithData:GETReply options:NSJSONReadingMutableLeaves|| NSJSONReadingMutableContainers error:nil];
   Nslog(@"%@",res);

ฉันคิดว่าคำตอบของคุณควรเป็นคำตอบที่ดีที่สุดเพราะดูเหมือนจะเป็นวิธีที่เร็วที่สุดในการเข้าถึงโครงสร้าง JSON
Porizm

2
ตัวเลือกไม่ควรใช้สอง | แต่เพียงอย่างเดียว | เนื่องจากพวกเขาจำเป็นต้องมีระดับบิต ORed
Deepak GM

คำถามไม่ได้ถามอะไรเกี่ยวกับคำขอเครือข่าย
โนอาห์กิลมอร์

2

@rckoenes แสดงวิธีรับข้อมูลของคุณอย่างถูกต้องจากสตริง JSON

สำหรับคำถามที่คุณถาม: EXC_BAD_ACCESSมักจะเกิดขึ้นเมื่อคุณพยายามเข้าถึงวัตถุหลังจากที่ปล่อย [อัตโนมัติ -] สิ่งนี้ไม่เฉพาะเจาะจงกับ JSON [de-] การทำให้เป็นอันดับ แต่จะทำอย่างไรกับการที่คุณได้รับออบเจ็กต์จากนั้นเข้าถึงมันหลังจากที่มันถูกปล่อยออกมา ความจริงที่ว่ามันผ่านทาง JSON นั้นไม่สำคัญ

มีหน้ามากมายที่อธิบายวิธีการแก้ไขปัญหานี้ - คุณต้องการ Google (หรือ SO) obj-c zombie objectsและโดยเฉพาะอย่างยิ่งNSZombieEnabledซึ่งจะพิสูจน์คุณค่าให้คุณในการช่วยกำหนดแหล่งที่มาของวัตถุซอมบี้ของคุณ ("ซอมบี้" คือสิ่งที่เรียกว่าเมื่อคุณปล่อยวัตถุ แต่ให้ตัวชี้ไปที่มันและลองอ้างอิงในภายหลัง)


1

Swift 2.0 บน Xcode 7 (Beta) พร้อมกับ do / try / catch block:

// MARK: NSURLConnectionDataDelegate

func connectionDidFinishLoading(connection:NSURLConnection) {
  do {
    if let response:NSDictionary = try NSJSONSerialization.JSONObjectWithData(receivedData, options:NSJSONReadingOptions.MutableContainers) as? Dictionary<String, AnyObject> {
      print(response)
    } else {
      print("Failed...")
    }
  } catch let serializationError as NSError {
    print(serializationError)
  }
}

1

หมายเหตุ: สำหรับสวิฟท์ 3 สตริง JSON ของคุณกำลังส่งคืนอาร์เรย์แทนที่จะเป็นพจนานุกรม โปรดลองสิ่งต่อไปนี้:

        //Your JSON String to be parsed
        let jsonString = "[{\"id\": \"1\", \"name\":\"Aaa\"}, {\"id\": \"2\", \"name\":\"Bbb\"}]";

        //Converting Json String to NSData
        let data = jsonString.data(using: .utf8)

        do {

            //Parsing data & get the Array
            let jsonData = try JSONSerialization.jsonObject(with: data!, options: .allowFragments) as! [AnyObject]

            //Print the whole array object
            print(jsonData)

            //Get the first object of the Array
            let firstPerson = jsonData[0] as! [String:Any]

            //Looping the (key,value) of first object
            for (key, value) in firstPerson {
                //Print the (key,value)
                print("\(key) - \(value) ")
            }

        } catch let error as NSError {
            //Print the error
            print(error)
        }

0
#import "homeViewController.h"
#import "detailViewController.h"

@interface homeViewController ()

@end

@implementation homeViewController

- (id)initWithStyle:(UITableViewStyle)style
{
    self = [super initWithStyle:style];
    if (self) {
        // Custom initialization
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    self.tableView.frame = CGRectMake(0, 20, 320, 548);
    self.title=@"Jason Assignment";

    // Uncomment the following line to preserve selection between presentations.
    // self.clearsSelectionOnViewWillAppear = NO;

    // Uncomment the following line to display an Edit button in the navigation bar for this view controller.
    // self.navigationItem.rightBarButtonItem = self.editButtonItem;
    [self clientServerCommunication];
}

-(void)clientServerCommunication
{
    NSURL *url = [NSURL URLWithString:@"http://182.72.122.106/iphonetest/getTheData.php"];
    NSURLRequest *req = [NSURLRequest requestWithURL:url];
    NSURLConnection *connection = [[NSURLConnection alloc]initWithRequest:req delegate:self];
    if (connection)
    {
        webData = [[NSMutableData alloc]init];
    }
}
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
    [webData setLength:0];
}

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

- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    NSDictionary *responseDict = [NSJSONSerialization JSONObjectWithData:webData options:0 error:nil];

    /*Third party API
     NSString *respStr = [[NSString alloc]initWithData:webData encoding:NSUTF8StringEncoding];
     SBJsonParser *objSBJson = [[SBJsonParser alloc]init];
     NSDictionary *responseDict = [objSBJson objectWithString:respStr]; */
    resultArray = [[NSArray alloc]initWithArray:[responseDict valueForKey:@"result"]];
    NSLog(@"resultArray: %@",resultArray);
    [self.tableView reloadData];
}


- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
//#warning Potentially incomplete method implementation.
    // Return the number of sections.
    return 1;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
//#warning Incomplete method implementation.
    // Return the number of rows in the section.
    return [resultArray count];
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"Cell";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
    if (cell == nil) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:CellIdentifier];
    }

    // Configure the cell...
    cell.textLabel.text = [[resultArray objectAtIndex:indexPath.row] valueForKey:@"name"];
    cell.detailTextLabel.text = [[resultArray objectAtIndex:indexPath.row] valueForKey:@"designation"];

    NSData *imageData = [NSData dataWithContentsOfURL:[NSURL URLWithString:[[resultArray objectAtIndex:indexPath.row] valueForKey:@"image"]]];
cell.imageview.image = [UIImage imageWithData:imageData];

    return cell;
}

/*
// Override to support conditional editing of the table view.
- (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath
{
    // Return NO if you do not want the specified item to be editable.
    return YES;
}
*/

/*
// Override to support editing the table view.
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath
{
    if (editingStyle == UITableViewCellEditingStyleDelete) {
        // Delete the row from the data source
        [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
    }   
    else if (editingStyle == UITableViewCellEditingStyleInsert) {
        // Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view
    }   
}
*/

/*
// Override to support rearranging the table view.
- (void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)fromIndexPath toIndexPath:(NSIndexPath *)toIndexPath
{
}
*/

/*
// Override to support conditional rearranging of the table view.
- (BOOL)tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath
{
    // Return NO if you do not want the item to be re-orderable.
    return YES;
}
*/


#pragma mark - Table view delegate

// In a xib-based application, navigation from a table can be handled in -tableView:didSelectRowAtIndexPath:
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    // Navigation logic may go here, for example:
     //Create the next view controller.
    detailViewController *detailViewController1 = [[detailViewController alloc]initWithNibName:@"detailViewController" bundle:nil];

 //detailViewController *detailViewController = [[detailViewController alloc] initWithNibName:@"detailViewController" bundle:nil];

 // Pass the selected object to the new view controller.

 // Push the view controller.
 detailViewController1.nextDict = [[NSDictionary alloc]initWithDictionary:[resultArray objectAtIndex:indexPath.row]];
 [self.navigationController pushViewController:detailViewController1 animated:YES];

    // Pass the selected object to the new view controller.

    // Push the view controller.
  //  [self.navigationController pushViewController:detailViewController animated:YES];
}



@end

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view from its nib.
    empName.text=[nextDict valueForKey:@"name"];
    deptlbl.text=[nextDict valueForKey:@"department"];
    designationLbl.text=[nextDict valueForKey:@"designation"];
    idLbl.text=[nextDict valueForKey:@"id"];
    salaryLbl.text=[nextDict valueForKey:@"salary"];
    NSString *ImageURL = [nextDict valueForKey:@"image"];
    NSData *imageData = [NSData dataWithContentsOfURL:[NSURL URLWithString:ImageURL]];
    image.image = [UIImage imageWithData:imageData];
}

0

ดูเหมือนว่าปัญหาจะเกิดขึ้นกับการยกเลิกการวางตำแหน่งวัตถุ NSJSONSerialization JSONObjectWithData เห็นได้ชัดว่ากำลังสร้างออบเจกต์ autoreleased บางส่วนและส่งคืนให้คุณ หากคุณพยายามที่จะนำไปใช้กับเธรดอื่นมันจะไม่ทำงานเนื่องจากไม่สามารถยกเลิกการจัดสรรคืนบนเธรดอื่น

เคล็ดลับอาจลองทำสำเนาที่ไม่แน่นอนของพจนานุกรมหรืออาร์เรย์นั้นและใช้มัน

NSError *e = nil;
id jsonObject = [NSJSONSerialization 
JSONObjectWithData: data 
options: NSJSONReadingMutableContainers 
error: &e] mutableCopy];

การรักษา NSDictionary เป็น NSArray จะไม่ส่งผลให้เกิดข้อยกเว้นการเข้าถึงที่ไม่ดี แต่อาจจะล้มเหลวเมื่อทำการเรียกเมธอด

นอกจากนี้อาจเป็นตัวเลือกที่ไม่สำคัญที่นี่ แต่เป็นการดีกว่าที่จะให้ NSJSONReadingMutableContainers | NSJSONReadingMutableContainers | NSJSONReadingAllowFragments แต่แม้ว่าพวกเขาจะเป็นวัตถุอัตโนมัติมันอาจไม่แก้ปัญหานี้


Deepak คุณแสดงรายการ NSJSONReadingMutableContainers สองครั้ง คุณหมายถึงหนึ่งสำหรับ NSJSONReadingMutableLeaves หรือไม่
jk7

0

ตัวอย่างที่ไม่ดีควรเป็นเช่นนี้ {"id": 1, "name": "something as name"}

จำนวนและสตริงที่ผสม

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.