IPhone पर JSON और कोर डेटा


93

मेरे पास एक मुख्य डेटा ऑब्जेक्ट ग्राफ है (जिसमें कई संबंधों से जुड़े दो निकाय शामिल हैं)।

मैं एक अपेक्षाकृत अनुभवहीन iPhone डेवलपर के रूप में उत्सुक था, चाहे कोई भी दृष्टिकोण की सिफारिश कर सकता हो, और iPhone के लिए एक उपयुक्त JSON कार्यान्वयन, जो मुझे इसकी अनुमति देगा:

  1. कोर डेटा रिकॉर्ड को JSON स्ट्रिंग में परिवर्तित करें (संस्थाओं के बीच संबंध बनाए रखने के दौरान); तथा

  2. JSON स्ट्रिंग को वापस कोर डेटा ऑब्जेक्ट्स में परिवर्तित करें (फिर से संस्थाओं के बीच संबंध को संरक्षित करते हुए)।

मैंने इस बिंदु पर एक ट्यूटोरियल / कोड नमूने के लिए असफल रूप से खोज की है, इसलिए किसी भी सहायता को कृतज्ञता प्राप्त होगी।


3
NSJSONSerialization
IOS5 के

मुझे पता है कि यह प्रश्न थोड़ा पुराना है, लेकिन मैंने OSReflectionKit नामक एक साधारण पुस्तकालय बनाया है , जो आपको NSJSONSerialization , या NSDictionary का उपयोग करके JSON से / से ऑब्जेक्ट्स को क्रमांकित / deserialize करने की अनुमति देता है । यह कोर डेटा ऑब्जेक्ट का भी समर्थन करता है।
अलेक्जेंड्रे OS

जवाबों:


103

सबसे पहले, एक JSON लाइब्रेरी का उपयोग करने के लिए चुनें, मुझे व्यक्तिगत रूप से TouchJSON पसंद है लेकिन कई अन्य बाहर भी काफी अच्छे हैं। जटिल भाग, हालांकि बहुत कठिन नहीं है, रूपांतरण के लिए अपनी प्रबंधित वस्तुओं को उपयुक्त संरचनाओं में बदलना है। मैंने यह वास्तविक त्वरित लिखा है, इसलिए इसमें त्रुटि या दो :) हो सकते हैं

आपके द्वारा कॉल की जाने वाली विधियाँ हैं:

- (NSString*)jsonStructureFromManagedObjects:(NSArray*)managedObjects;
- (NSArray*)managedObjectsFromJSONStructure:(NSString*)json withManagedObjectContext:(NSManagedObjectContext*)moc;

और कार्यान्वयन इस प्रकार है:

- (NSDictionary*)dataStructureFromManagedObject:(NSManagedObject*)managedObject
{
  NSDictionary *attributesByName = [[managedObject entity] attributesByName];
  NSDictionary *relationshipsByName = [[managedObject entity] relationshipsByName];
  NSMutableDictionary *valuesDictionary = [[managedObject dictionaryWithValuesForKeys:[attributesByName allKeys]] mutableCopy];
  [valuesDictionary setObject:[[managedObject entity] name] forKey:@"ManagedObjectName"];
  for (NSString *relationshipName in [relationshipsByName allKeys]) {
    NSRelationshipDescription *description = [[[managedObject entity] relationshipsByName] objectForKey:relationshipName];
    if (![description isToMany]) {
      NSManagedObject *relationshipObject = [managedObject valueForKey:relationshipName];
      [valuesDictionary setObject:[self dataStructureForManagedObject:relationshipObject] forKey:relationshipName];
      continue;
    }
    NSSet *relationshipObjects = [managedObject objectForKey:relationshipName];
    NSMutableArray *relationshipArray = [[NSMutableArray alloc] init];
    for (NSManagedObject *relationshipObject in relationshipObjects) {
      [relationshipArray addObject:[self dataStructureForManagedObject:relationshipObject]];
    }
    [valuesDictionary setObject:relationshipArray forKey:relationshipName];
  }
  return [valuesDictionary autorelease];
}

- (NSArray*)dataStructuresFromManagedObjects:(NSArray*)managedObjects
{
  NSMutableArray *dataArray = [[NSMutableArray alloc] init];
  for (NSManagedObject *managedObject in managedObjects) {
    [dataArray addObject:[self dataStructureForManagedObject:managedObject]];
  }
  return [dataArray autorelease];
}

- (NSString*)jsonStructureFromManagedObjects:(NSArray*)managedObjects
{
  NSArray *objectsArray = [self dataStructuresFromManagedObjects:managedObjects];
  NSString *jsonString = [[CJSONSerializer serializer] serializeArray:objectsArray];
  return jsonString;
}

- (NSManagedObject*)managedObjectFromStructure:(NSDictionary*)structureDictionary withManagedObjectContext:(NSManagedObjectContext*)moc
{
  NSString *objectName = [structureDictionary objectForKey:@"ManagedObjectName"];
  NSManagedObject *managedObject = [NSEntityDescription insertNewObjectForEntityForName:objectName inManagedObjectContext:moc];
  [managedObject setValuesForKeysWithDictionary:structureDictionary];

  for (NSString *relationshipName in [[[managedObject entity] relationshipsByName] allKeys]) {
    NSRelationshipDescription *description = [relationshipsByName objectForKey:relationshipName];
    if (![description isToMany]) {
      NSDictionary *childStructureDictionary = [structureDictionary objectForKey:relationshipName];
      NSManagedObject *childObject = [self managedObjectFromStructure:childStructureDictionary withManagedObjectContext:moc];
      [managedObject setObject:childObject forKey:relationshipName];
      continue;
    }
    NSMutableSet *relationshipSet = [managedObject mutableSetForKey:relationshipName];
    NSArray *relationshipArray = [structureDictionary objectForKey:relationshipName];
    for (NSDictionary *childStructureDictionary in relationshipArray) {
      NSManagedObject *childObject = [self managedObjectFromStructure:childStructureDictionary withManagedObjectContext:moc];
      [relationshipSet addObject:childObject];
    }
  }
  return managedObject;
}

- (NSArray*)managedObjectsFromJSONStructure:(NSString*)json withManagedObjectContext:(NSManagedObjectContext*)moc
{
  NSError *error = nil;
  NSArray *structureArray = [[CJSONDeserializer deserializer] deserializeAsArray:json error:&error];
  NSAssert2(error == nil, @"Failed to deserialize\n%@\n%@", [error localizedDescription], json);
  NSMutableArray *objectArray = [[NSMutableArray alloc] init];
  for (NSDictionary *structureDictionary in structureArray) {
    [objectArray addObject:[self managedObjectFromStructure:structureDictionary withManagedObjectContext:moc]];
  }
  return [objectArray autorelease];
}

अब यह पुनरावर्ती है, इसलिए यदि आप सावधान नहीं हैं तो आप आसानी से अपने संपूर्ण स्टोर का अनुवाद कर सकते हैं। अपने रिश्तों को देखें और सुनिश्चित करें कि वे केवल ऑब्जेक्ट ट्री को "डाउन" करते हैं ताकि आप केवल उन वस्तुओं को प्राप्त कर सकें जिन्हें आप अनुवादित करना चाहते हैं।


एक और उत्कृष्ट उत्तर के लिए और आपकी बहुत उपयोगी पुस्तक के लिए फिर से धन्यवाद! :)
२१:११

2
हाय माक्र्स। मैंने अभी ऊपर कोड की कोशिश की है (कुछ छोटे संशोधनों के साथ इसे संकलित करने के लिए और निष्पादन अनिश्चित रूप से ऐप क्रैश होने तक लगता है)। आपको परेशान करने के लिए क्षमा करें, लेकिन मैं उत्सुक था कि क्या आप शायद इस समस्या को हल करने के लिए मुझे सही दिशा में इंगित कर सकते हैं। यह datastructureFromManagedObject पद्धति में पुनरावृत्ति के साथ होता है ...
Urizen

1
आपके डेटा संरचना पर निर्भर करता है। यदि आपका मॉडल लूप का उत्पादन करेगा तो यह हमेशा के लिए चलेगा। अपने डेटा मॉडल की समीक्षा करें और यह सुनिश्चित करें कि यह एक पेड़ का डिज़ाइन है या लूपिंग को रोकने के लिए पुनरावर्ती कोड में लॉजिक स्टॉप लगाता है।
मार्कस एस। ज़रा

1
क्या आपने वास्तव में इस कोड को चलाने की कोशिश की है? बहुत सारी त्रुटियाँ हैं। dataStructureForManagedObject भी मौजूद नहीं है। मैंने सोचा कि यह सिर्फ एक टाइपो हो सकता है, लेकिन अगर आप इसे डेटास्ट्रीमक्चर में बदल देते हैं, तो यह पूरी तरह से पूरी तरह से हटा दें, बस रिश्ते जोड़े के बीच असीम रूप से उछलता है। क्या मुझे यहाँ कुछ अतिरिक्त कोड याद आ रहे हैं?
क्रिस मिचेलमोर

1
यह कोड उदाहरण दो साल पहले ब्राउज़र में लिखा गया था। यह एक कॉपी और पेस्ट न होने के लिए प्रेरित करने का इरादा था। एक अनंत लूप के रूप में, इसका मतलब है कि आपके मॉडल में एक लूप है और आपको चक्र को तोड़ने के लिए अपने ऐप में मॉडल विशिष्ट तर्क जोड़ना होगा। ऐसा करने के कई तरीके हैं जो इस उदाहरण में नहीं हैं।
मार्कस एस। ज़रा

12

मैं बस एक छोटे टाइपो को इंगित करना चाहता था, जिससे कोड दुर्घटनाग्रस्त हो गया, और उम्मीद है कि यह आपको कुछ मिनट बचाएगा।

- (NSArray*)dataStructuresFromManagedObjects:(NSArray*)managedObjects {

    NSMutableArray *dataArray = [[NSArray alloc] init];
    for (NSManagedObject *managedObject in managedObjects) {
        [dataArray addObject:[self dataStructureFromManagedObject:managedObject]];
    }
    return [dataArray autorelease];
}

NSMutableArray *dataArray = [[NSArray alloc] init]; // This should be NSMutableArray

वास्तव में होना चाहिए NSMutableArray *dataArray = [[NSMutableArray alloc] init];

वह सब है।

धन्यवाद


10

रेल के साथ कोर डेटा को सिंक्रनाइज़ करना एक विस्तृत प्रस्तुति है जिसमें जेन्सन से / से अपने कोर डेटा ऑब्जेक्ट्स को क्रमांकित / डीसर्लाइज़ करने के लिए नमूना कोड शामिल है (कोर डेटा भाग के लिए 55 को स्लाइड करने के लिए छोड़ें)। उनका नमूना कोड रिश्तों के बिना काफी सरल मॉडल मानता है, हालांकि मुझे लगता है कि इसका विस्तार करना बहुत आसान होगा।

प्रस्तुति आपके कोर डेटा मॉडल को REST- आधारित वेब एप्लिकेशन के साथ सिंक में रखने के बारे में कुछ विस्तार से बताती है , जिसमें ऑब्जेक्ट्स के साथ कुछ उपयोगी लाइब्रेरी शामिल हैं, जिसमें ObjectiveResource और ASIHTTPRequest शामिल हैं । निश्चित नहीं है कि आप क्या करने की कोशिश कर रहे हैं, लेकिन यह कोर डेटा कोड के लिए भी देखने लायक है।


7

यदि आपके पास NSDateअपनी प्रबंधित वस्तु है, जैसा कि टिप्पणियों में से एक में ऊपर उल्लेख किया गया है, तो आपको ऑब्जेक्ट को शामिल करने में समस्या होगी NSDate। एक साधारण फिक्स उद्देश्य-सी श्रेणियों का उपयोग करने के लिए एक JSONDataRepresentationविधि जोड़ना है NSDate

इन दो फ़ाइलों को अपनी परियोजना में जोड़ें:

NSdate.h:

#import <Foundation/Foundation.h>

@interface NSDate (jsondatarepresentation) 

- (NSData*) JSONDataRepresentation;

@end

NSDate.m:

#import "NSDate.h"

@implementation NSDate (jsondatarepresentation)

- (NSData*) JSONDataRepresentation {
    return [[[NSNumber numberWithDouble:[self timeIntervalSince1970]] stringValue] dataUsingEncoding:NSUTF8StringEncoding];
}

@end


2

मैं इस पोस्ट पर आया जो बहुत अच्छा काम करता है।

http://touchalicious.com/blog/2009/10/25/turn-core-data-models-into-json.html

चूंकि यह पुनरावर्ती है, इसलिए कई-से-कई रिश्ते खुद के माध्यम से लूपिंग रखने जा रहे हैं। इससे बचने के लिए, मैंने अपने Core Data मॉडल में रिश्तों की उपयोगकर्ता जानकारी शब्दकोश में ".Exportable" कुंजी जोड़ी। फिर आप इस कुंजी की जांच कर सकते हैं और इसके बिना संबंधों के माध्यम से लूप नहीं चुन सकते हैं।

यहां छवि विवरण दर्ज करें

if ([property isKindOfClass:[NSRelationshipDescription class]])
    {
        NSRelationshipDescription *relationshipDescription = (NSRelationshipDescription *)property;

        if ([[[relationshipDescription userInfo] objectForKey:@"isExportable"] boolValue] == YES)
        {
            NSString *name = [relationshipDescription name];

            if ([relationshipDescription isToMany])
            {
                NSMutableArray *arr = [properties valueForKey:name];
                if (!arr)
                {
                    arr = [[NSMutableArray alloc] init];
                    [properties setValue:arr forKey:name];
                }

                for (NSManagedObject *o in [self mutableSetValueForKey:name])
                {
                    [arr addObject:[o propertiesDictionary]];
                }
            }
            else
            {
                NSManagedObject *o = [self valueForKey:name];
                [properties setValue:[o propertiesDictionary] forKey:name];
            }
        }
    }
}

2

बस सोचा आईडी इस सवाल के लिए एक त्वरित अद्यतन पोस्ट। मैंने माक्र्स और ब्रैंडन के उत्तरों का पालन किया और JSON निर्यात के लिए इसे लेकर आया (यह अभी भी TouchJSON का उपयोग करता है):

- (NSData*)jsonStructureFromManagedObjects:(NSArray*)managedObjects
{
    NSArray *objectsArray = [self dataStructuresFromManagedObjects:managedObjects];
    NSData *jsonData      = [[CJSONSerializer serializer] serializeArray:objectsArray error:nil];
    return jsonData;
}

- (NSArray*)dataStructuresFromManagedObjects:(NSArray*)managedObjects
{
    NSMutableArray *dataArray = [[NSMutableArray alloc] init];
    for (NSManagedObject *managedObject in managedObjects) {
        [dataArray addObject:[self dataStructureFromManagedObject:managedObject]];
    }
    return dataArray;
}

- (NSDictionary*)dataStructureFromManagedObject:(NSManagedObject*)managedObject
{
    NSDictionary *attributesByName        = [[managedObject entity] attributesByName];
    NSDictionary *relationshipsByName     = [[managedObject entity] relationshipsByName];
    NSMutableDictionary *valuesDictionary = [[managedObject dictionaryWithValuesForKeys:[attributesByName allKeys]] mutableCopy];
    [valuesDictionary setObject:[[managedObject entity] name] forKey:@"ManagedObjectName"];

    for (NSString *relationshipName in [relationshipsByName allKeys]) {

        NSRelationshipDescription *description = [[[managedObject entity] relationshipsByName] objectForKey:relationshipName];

        if ([[[description userInfo] objectForKey:@"isExportable"] boolValue] == YES) {

            if (![description isToMany]) {
                NSManagedObject *relationshipObject = [managedObject valueForKey:relationshipName];
                if (relationshipObject) {
                    [valuesDictionary setObject:[self dataStructureFromManagedObject:relationshipObject] forKey:relationshipName];
                }

                continue;
            }

            NSSet *relationshipObjects        = [managedObject valueForKey:relationshipName];
            NSMutableArray *relationshipArray = [[NSMutableArray alloc] init];

            for (NSManagedObject *relationshipObject in relationshipObjects) {
                [relationshipArray addObject:[self dataStructureFromManagedObject:relationshipObject]];
            }

            [valuesDictionary setObject:relationshipArray forKey:relationshipName];

        }

    }
    return valuesDictionary;
}

मैं आयात करने का काम नहीं कर सकता, हो सकता है कि इस तथ्य के साथ कुछ करना है कि मैं जादुई रिकॉर्ड का उपयोग कर रहा हूं, मुझे यकीन नहीं है, इसलिए इम बस आने वाली JSON स्ट्रीम के माध्यम से लूप कर रहा है और मैन्युअल रूप से ऑब्जेक्ट बना रहा है ...


1

मार्कस एस। ज़रा ने मुझे एक कार्यशील संस्करण के लिए पुनरावर्ती विचार लाने के लिए प्रेरित किया है। इस संस्करण में आपको CoreData में एक कुंजी सेट करने की आवश्यकता नहीं है और आप इसे अपनी परियोजना में काट सकते हैं और पेस्ट कर सकते हैं :-)

// MARK: - encoding and decoding CoreData entity to dictionary

func dataStructureFromManagedObject( managedObject:NSManagedObject?, parentEntity: NSEntityDescription? = nil) -> NSMutableDictionary {
    if (managedObject != nil) {
        var attributesByName: NSDictionary = managedObject!.entity.attributesByName
        var relationshipsByName: NSDictionary  = managedObject!.entity.relationshipsByName
        var valuesImmutableDictionary: NSDictionary = managedObject!.dictionaryWithValuesForKeys( attributesByName.allKeys)
        var valuesDictionary: NSMutableDictionary = valuesImmutableDictionary.mutableCopy() as NSMutableDictionary
        valuesDictionary.setObject( managedObject!.entity.name!, forKey: "ManagedObjectName")
        for relationshipNameObject in relationshipsByName.allKeys {
            var relationshipName: NSString = relationshipNameObject as  NSString
            var relationshipDescription: NSRelationshipDescription? = relationshipsByName.objectForKey( relationshipName) as? NSRelationshipDescription
            if !relationshipDescription!.toMany {
                // ono to one
                if parentEntity == nil || (relationshipDescription! as NSRelationshipDescription).destinationEntity != parentEntity! {
                    // no parent or relationship is "downward" -> object for relationship must be added
                    var relationshipObject: NSManagedObject? = managedObject!.valueForKey( relationshipName) as? NSManagedObject
                    var relationshipObjectDictionary: NSMutableDictionary = self.dataStructureFromManagedObject( relationshipObject, parentEntity: managedObject?.entity)
                    valuesDictionary.setObject( relationshipObjectDictionary, forKey: relationshipName)
                } else {
                    // relationship is "upward" -> nothing to do
                }
            } else {
                // one to many -> all objects must be added
                var relationshipObjects: NSSet = managedObject!.mutableSetValueForKey( relationshipName)
                var relationshipArray:NSMutableArray = []
                for relationshipObjectRaw in relationshipObjects {
                    var relationshipObject:NSManagedObject? = relationshipObjectRaw as? NSManagedObject
                    if relationshipObject != nil && !relationshipObject!.entity.isKindOfEntity( managedObject!.entity) {
                        relationshipArray.addObject(self.dataStructureFromManagedObject( relationshipObject, parentEntity: managedObject?.entity))
                    }
                }
                valuesDictionary.setObject( relationshipArray, forKey: relationshipName)
            }
        }
        return valuesDictionary
    } else {
        return NSMutableDictionary()
    }
}

func managedObjectFromStructure( structureDictionary: NSDictionary, moc: NSManagedObjectContext, parentObject: NSManagedObject? = nil) -> NSManagedObject {
    if structureDictionary.count > 0 {
        var objectName:NSString = structureDictionary.objectForKey( "ManagedObjectName") as NSString
        var managedObject:NSManagedObject = NSEntityDescription.insertNewObjectForEntityForName( objectName, inManagedObjectContext: moc) as NSManagedObject
        var relationshipsByName: NSDictionary  = managedObject.entity.relationshipsByName
        var realObjectStructure:NSMutableDictionary = structureDictionary.mutableCopy() as NSMutableDictionary
        realObjectStructure.removeObjectForKey( "ManagedObjectName")
        for key in realObjectStructure.allKeys {
            // search for "ManagedObjectName" relationship entrys and delete them before filling the managedObject from this structure
            for relationshipName in relationshipsByName.allKeys {
                if relationshipName as NSString == key as NSString {
                    realObjectStructure.removeObjectForKey( key)
                }
            }
        }
        managedObject.setValuesForKeysWithDictionary( realObjectStructure)
        // the main object with attributes is created. Now care about the relationships
        for relationshipName in managedObject.entity.relationshipsByName.keys {
            var description:NSRelationshipDescription = relationshipsByName.objectForKey( relationshipName) as NSRelationshipDescription
            if !description.toMany {
                // to one relationship
                if parentObject == nil || description.destinationEntity != parentObject!.entity {
                    // no parent or relationship is "downward" -> recurse structure to add
                    var childStructureDictionary:NSDictionary = structureDictionary.objectForKey( relationshipName) as NSDictionary
                    if childStructureDictionary.count > 0 {
                        // dictionary not empty -> object must be created and added
                        var childObject:NSManagedObject? = self.managedObjectFromStructure( childStructureDictionary, moc: moc, parentObject: managedObject)
                        // validateForUpdate
                        var error:NSError?
                        if !managedObject.validateForUpdate( &error) {
                            println("Error: Object not in valid state for update!!! -> \(error)")
                        } else {
                            managedObject.setValue( childObject, forKey: relationshipName as NSString)
                        }
                    } else {
                        // relationship is "upward" -> nothing to do
                    }
                }
            } else {
                // to many relationship
                var relationshipSet:NSMutableSet = managedObject.mutableSetValueForKey( relationshipName as NSString)
                var relationshipArray:NSArray = structureDictionary.objectForKey( relationshipName as NSString) as NSArray
                for childStructureDictionary in relationshipArray {
                    if childStructureDictionary.count > 0 {
                        // dictionary not empty -> object must be created and added
                        var childObject:NSManagedObject = self.managedObjectFromStructure( childStructureDictionary as NSDictionary, moc: moc, parentObject: managedObject)
                        // validateForUpdate
                        var error:NSError?
                        if !managedObject.validateForUpdate( &error) {
                            println( "Error: Object not in valid state for update!!! -> \(error)")
                        } else {
                            relationshipSet.addObject( childObject)
                        }
                    } else {
                        // no object was behind the relationship -> nothing to do
                    }
                }
                // save set
                managedObject.setValue( relationshipSet, forKey: relationshipName as NSString)
            }
        }
        // final check validateForUpdate
        var error:NSError?
        if !managedObject.validateForUpdate( &error) {
            println( "Error: Object not in valid state for update although all previous check are passed!!! -> \(error)")
        }
        return managedObject
    } else {
        println( "Error: structure for object was empty. this should not happen at this point")
        var objectName:NSString = structureDictionary.objectForKey( "ManagedObjectName") as NSString
        var managedObject:NSManagedObject = NSEntityDescription.insertNewObjectForEntityForName( objectName, inManagedObjectContext: moc) as NSManagedObject
        return managedObject
    }
}

func dataStructuresFromManagedObjects( managedObjects: NSArray) -> NSArray {
    var dataArray:NSMutableArray = []
    for managedObject in managedObjects {
        dataArray.addObject( self.dataStructureFromManagedObject(managedObject as? NSManagedObject))
    }
    return dataArray
}

यहां कुंजी पेरेंट एंटिटी को रिकर्सन के तर्क के रूप में पारित करना है, इसलिए हम यह तय कर सकते हैं कि हमें डेटा के साथ कौन सा संबंध भरना है। तो दोनों कार्य: dataStructureFromManagedObjectऔर managedObjectFromStructureकिसी भी ऑब्जेक्ट को CoreData से एक शब्दकोश में एन्कोड और डिकोड कर सकते हैं और एक ऑब्जेक्ट में वापस कर सकते हैं।

हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.