17

質問の概要:

、、、などのプロパティにSyncObjectKVCに準拠しているクラスについて考えてみます。timesomeValuelastChangeuuid

NSArrayのインスタンスのみを含むことを検討してくださいSyncObject

配列をJSON配列としてサーバーに送信する必要があります。

RestKitを使用したHTTPPOSTを使用して、この配列をサーバーに送信するにはどうすればよいですか?

配列の例:

[
  {
    "time": "14:45 10/21/2011",
    "someValue": "15",
    "lastChange": "14:45 10/21/2011",
    "uuid": "0b07c510-f4c8-11e0-be50-0800200c9a66"
  },
  {
    "time": "14:50 10/21/2011",
    "someValue": "62",
    "lastChange": "14:51 10/21/2011",
    "uuid": "1a6d4480-f4c8-11e0-be50-0800200c9a66"
  }
]

詳細

サーバーにJSONとして必要なオブジェクトの配列があります。RestKitがこれを行う最も簡単な方法のように思えます。オブジェクトをNSDictionaryオブジェクトのセットに変換し、JSONエンコーダーを使用POSTしてサーバーに送信できるJSONを取得することを避けようとしています。

したがって、配列を作成し、配列に格納されているオブジェクトのクラスのマッピングを設定したら、当然POST、サーバーに接続しようとします。

RKObjectManager* mgr = [RKObjectManager objectManagerWithBaseURL:@"http://localhost/someweb/api/"];
mgr.serializationMIMEType = RKMIMETypeFormURLEncoded;
mgr.client.username = @"username";
mgr.client.password = @"password";
RKObjectMapping* mapping = [RKObjectMapping mappingForClass:[NSMutableDictionary class]];
[mapping mapKeyPath: @"time"       toAttribute:@"time"         ];  
[mapping mapKeyPath: @"someValue"  toAttribute:@"someValue"    ];
[mapping mapKeyPath: @"lastChange" toAttribute:@"lastChange"   ];
[mapping mapKeyPath: @"uuid"       toAttribute:@"uuid"         ];
RKObjectMapping* mappingForSerialization = [mapping inverseMapping];
[mgr.mappingProvider setSerializationMapping:mappingForSerialization 
                                    forClass:[NSManagedObject class]];

[mgr.router routeClass:[NSManagedObject class] toResourcePath:@"/sync" forMethod:RKRequestMethodPOST];    

[mgr postObject:array delegate:nil/*self*/];

しかし、これは私が得たものです:

2011-10-11 14:57:51.769 AppConnect [1974:6e0b] ***キャッチされなかった例外のためにアプリを終了しています'(null)'、理由:'HTTPメソッドのタイプ'__NSArrayI'のオブジェクトのルーティング可能なパスが見つかりません'役職''

どうやら、RestKitはNSArraysの処理方法を知りません。

RestKitを使用してオブジェクトの配列を投稿するにはどうすればよいですか?


私は別のことを試しました。最後の行を手動の送信に置き換えましたRKObjectLoader

//[mgr postObject:[NSMutableDictionary dictionaryWithObject:array forKey:@"data"] delegate:nil/*self*/];


NSString* syncPath = @"/sync"; 
RKObjectLoader * objectLoader = [mgr objectLoaderWithResourcePath:syncPath delegate:self];
objectLoader.serializationMIMEType = RKMIMETypeJSON;
objectLoader.method = RKRequestMethodPOST;
//objectLoader.objectClass = [NSManagedObject class];       
//objectLoader.managedObjectStore = mgr.objectStore; 
objectLoader.params = [NSDictionary dictionaryWithObject:array 
                                                  forKey:@"MyData"]; 
[objectLoader send];

残念ながら、これはどのような種類のマッピングも適用せず、代わりにオブジェクトの説明の配列を送信します。設定serializationMIMETypeも送信内容の構造には影響せず、params常にとして送信されapplication/x-www-form-urlencodedます。

また、シリアル化マッピングを割り当てて、オブジェクトをtargetObjectandとして渡してみましたsourceObject(これは、RestKitが内部で行うことのようです-[RKObjectManager postObject:delegate:])。

RKObjectLoader * objectLoader = [mgr objectLoaderWithResourcePath:syncPath delegate:self];
objectLoader.method = RKRequestMethodPOST;
//objectLoader.objectClass = [NSManagedObject class];       
//objectLoader.managedObjectStore = mgr.objectStore; 
objectLoader.params = [NSMutableDictionary dictionaryWithObject:array 
                                                  forKey:@"MyData"]; 
objectLoader.serializationMapping = mapping;
objectLoader.serializationMIMEType = RKMIMETypeJSON;
objectLoader.sourceObject = objectLoader.params;
objectLoader.targetObject = objectLoader.params;
[objectLoader send];

残念ながら、この方法ではマッピングは行われません。

2011-10-12 12:36:48.143 MyProject[5119:207] D restkit.network:RKObjectLoader.m:290 POST or PUT request for source object {
    MyData =     (
        "<NSManagedObject: 0x5935430> (entity: SomeRecord; id: 0x5934da0 <x-coredata://64DF9977-DA50-4FCD-8C20-4132E58439BF/SomeRecord/p1> ; data: <fault>)",
        "<NSManagedObject: 0x5935730> (entity: SomeRecord; id: 0x5934db0 <x-coredata://64DF9977-DA50-4FCD-8C20-4132E58439BF/SomeRecord/p2> ; data: <fault>)"
    );
}, serializing to MIME Type application/json for transport...
2011-10-12 12:36:48.143 MyProject[5119:207] D restkit.object_mapping:RKObjectMappingOperation.m:428 Starting mapping operation...
2011-10-12 12:36:48.145 MyProject[5119:207] T restkit.object_mapping:RKObjectMappingOperation.m:291 Did not find mappable attribute value keyPath 'time'
2011-10-12 12:36:48.145 MyProject[5119:207] T restkit.object_mapping:RKObjectMappingOperation.m:291 Did not find mappable attribute value keyPath 'someValue'
2011-10-12 12:36:48.145 MyProject[5119:207] T restkit.object_mapping:RKObjectMappingOperation.m:291 Did not find mappable attribute value keyPath 'lastChange'
2011-10-12 12:36:48.145 MyProject[5119:207] T restkit.object_mapping:RKObjectMappingOperation.m:291 Did not find mappable attribute value keyPath 'uuid'
2011-10-12 12:36:48.145 MyProject[5119:207] D restkit.object_mapping:RKObjectMappingOperation.m:448 Mapping operation did not find any mappable content
2011-10-12 12:36:48.146 MyProject[5119:207] T restkit.network:RKRequest.m:211 Prepared POST URLRequest '<NSMutableURLRequest http://someurl/api/sync?request=provide_key>'. HTTP Headers: {
    Accept = "application/json";
    "Content-Length" = 0;
}. HTTP Body: .
4

2 に答える 2

18

クラスのルーティングを定義したため、restkitはNSArrayのルーティング可能なパスに資金を提供しませんNSManagedObjectMySyncEntityマッピングで定義したivarを保持するカスタムクラスを作成することをお勧めします。次に、次のようにマッピングを作成します。

RKObjectMapping* mapping = [RKObjectMapping mappingForClass:[MySyncEntity class]];
....
[myManager setSerializationMIMEType:RKMIMETypeJSON];
[[myManager router] routeClass:[MySyncEntity class] toResourcePath:@"/sync"];

そうすれば、オブジェクトをJSONオブジェクトとしてAPIバックエンドに投稿できるはずです。

さらなる解明:

この場合、NSManagedObjectインスタンスの配列をJSONベースのAPIに投稿します。これを行うには、オブジェクトを配列に保持する同期エンティティを作成する必要があります。

@interface MySyncEntity : NSObject {}
@property (nonatomic, retain) NSArray* mySyncArray;
...
@end 

mySyncArray、残りのバックエンドに送信するペイロードを保持します。NSManagedObject次に、送信されるものmySyncArrayMySyncEntityエンティティ自体の両方に適切なマッピングを作成します。

RKObjectManager *manager = [RKObjectManager objectManagerWithBaseURL:kBaseUrl];
...
RKObjectMapping *mngObjMapping = [RKObjectMapping mappingForClass:[NSManagedObject class]]; 
[mngObjMapping mapKeyPath: @"time" toAttribute:@"time"];  
[mngObjMapping mapKeyPath: @"recordLevel" toAttribute:@"recordLevel"];
.... //map as many properties as you wish
[[manager mappingProvider] setSerializationMapping:[mngObjMapping inverseMapping]
                                          forClass:[NSManagedObject class]];

//now, we create mapping for the MySyncEntity
RKObjectMapping *syncEntityMapping = [RKObjectMapping mappingForClass:[MySyncEntity class]];
[syncEntityMapping mapKeyPath:@"mySyncArray" toRelationship:@"mySyncArray" withMapping:mngObjMapping];
[[manager mappingProvider] setSerializationMapping:[syncEntityMapping inverseMapping]
                                          forClass:[MySyncEntity class]];

マッピングが定義されたら、オブジェクトをサーバーに投稿できます

[manager postObject:mySyncInstance delegate:nil];

配列の内容は、定義されたRESTエンドポイントmySyncInstanceに従ってマッピングされて送信されます。mngObjMapping

于 2011-10-11T13:30:20.530 に答える
3

さらに明確にするために、mjaの回答で重要なことは

[syncEntityMapping mapKeyPath:@"mySyncArray" toRelationship:@"mySyncArray" withMapping:mngObjMapping];

これは、「mySyncArrayキーパスは、に従ってマップする必要があるオブジェクトを含む配列です」と言いますmngObjMapping

于 2011-10-12T13:50:58.943 に答える