iOS: Easy way of accessing XML data records?
I have an application which needs to load data from a remote server (businesses by location) and display them on a map. The feed from the server is XML based. I have successfully done an implementation of this but wondered if I could make my life a bit easier.
Currently, I am using Google's GDataXML library, I have implemented a library for getting remote data in the background and it calls me back when complete or during loading if I want.
When the full data has been loaded, I traverse the document and convert the different levels to be an object, adding that object to an NSMutableArray (as I want to do lazy loading so I want to add as more is requested) and then passing that array to the next bit of my app that then interprets and pins/annotates the map for me.
Example XML data (abstracted):
<businesses>
  <business>
    <name> Fred Bloggs and Co </name>
    <address> 123 No Street, Nowhere </address>
    <town> Somesville </town>
    <county> Someshire </county>
    <postcode> XX11 1XX </postcode>
  </business>
  ..... more records but you get the idea .....
</businesses>
Example storage object (abstracted)
-- businessrecord.h --
@interface BusinessRecord : NSObject {
    NSString *name;
    NSString *address;
    NSString *town;
    NSString *county;
    NSString *postcode;
}
@property (nonatomic, copy) 开发者_Go百科NSString *name;
@property (nonatomic, copy) NSString *address;
@property (nonatomic, copy) NSString *town;
@property (nonatomic, copy) NSString *county;
@property (nonatomic, copy) NSString *postcode;
@end
-- businessrecord.m --
@implementation BusinessRecord
@synthesize name, address, town, county, postcode;
@end
Right, so you can probably guess that i'm manually parsing each XML element in each node and manually transferring them to the BusinessRecord object.
This is particularly time consuming as I have to write three or more lines of code in different places for each property.
Under GDataXML you access each element with something like:
NSArray *businesses = [[[xmlRoot elementsForName:@"businesses"] objectAtIndex:0] elementsForName:@"business"];
NSMutableArray *businessList = [[NSMutableArray alloc] initWithCapacity: businesses.count];
for (int i=0; i<businesses.count; i++) {
    GDataXMLElement *business = [businesses objectAtIndex: i];
    BusinessRecord *busRec = [[BusinessRecord alloc] init];
    busRec.name = [[[bus elementsForName:@"name"] objectAtIndex:0] stringValue];
    .... etc for each element ...
    [businessList addObject: busRec];
    [busRec release];
}
This all seems extremely long winded to me, there must be a better way of doing this?
What I want to end up with is everything in my XML at the level "business" in some kind of random access array, I don't want to individually specify each element.
Ideally, something like in PHP where you can just have a sequential array containing associative arrays like:
$businessList = Array(
  Array(
    "name" => "Fred Bloggs and Co",
    "address" => "123 No Street",
    "town" => "Sometown",
    "county" => "Someshire",
    "postcode" => "XX11 1XX"
  )
);
So i'm guessing I need to write a wrapper class that can interpret a specific element and get all its sub elements right (ie, enumerate the elements and then work on them).
What do I store this in? NSDictionary?
Maybe something like (pseudo code):
NSArray *businesses = [[[xmlRoot elementsForName:@"businesses"] objectAtIndex:0] elementsForName:@"business"];
NSMutableArray *businessList = [[NSMutableArray alloc] init];
[xmlBreakerUpper breakUpXMLAtElement: businesses addToMutableArray: &businessList];
Has anyone any experience with GDataXML and can help me with enumeration as I can't follow the sparse documentation.
I'm not totally attached to GDataXML at this stage, i've only two classes that count on it so if there's a better way.
I am in control of the server output but the client has a preference for XML for other data access API's they may or may not want to implement later.
I don't need to send XML back to the server at this stage, the requests are HTTP GET based at present but may become POST later when people fill in forms.
All help or pushes in the right direction greatfully received.
Option 1:
You could create your own init method for your businessrecord class that accepts a GDataXMLElement argument and parse out the values you need using some XPath expressions, which are a bit more pleasant on the eyes ;)
Option 2:
To achieve the PHP-like array of associative arrays, you can: Iterate over the array of "business" elements, and iterate over each business element's children, using the child element names as keys for your NSMutableDictionary business record - and stash the newly created dictionary in the businessList array.
/* THIS IS JUST A QUICK PSEUDO-CODE-ISH SAMPLE */
NSArray *businesses = [xmlRoot nodesForXPath:@"//businesses/business" error:nil];
NSMutableArray *businessList = [[NSMutableArray alloc] initWithCapacity: businesses.count];
GDataXMLElement *business;
for (business in businesses)
{
    NSMutableDictionary *businessRecord = [[NSMutableDictionary] array];
    // iterate over child elements of "business", use element names as keys for businessRecord
    NSArray *businessChildren = [business children];    
    GDataXMLElement *businessChild;
    for (businessChild in businessChildren)
    {
        [businessRecord setValue:[businessChild value] forKey:[businessChild name]]
    }
    [businessList addObject:businessRecord];
    [businessRecord release];
}
I stumbled upon this article "How To Read and Write XML Documents with GDataXML" while researching options for XML parsing that was a solid read, however they are parsing in specific model objects, rather than a more generic data object like you're looking for.
You should check out their other article "How To Choose The Best XML Parser For Your iPhone Project" as GDataXML seems like overkill.
 
         加载中,请稍侯......
 加载中,请稍侯......
      
精彩评论