【问题标题】:Online and offline support for RestKit iOS applications对 RestKit iOS 应用程序的在线和离线支持
【发布时间】:2012-03-01 01:20:16
【问题描述】:

我想将 RestKit 用于需要同时在线和离线工作的应用程序。

我可能误解了 RestKit 的工作原理,但我认为这(相当)容易实现。

在一个临时测试 iOS 应用程序中,我进行了如下设置:

// setup the client
NSString* URL = @"http://10.211.55.5:3000/api";
RKClient* client = [RKClient clientWithBaseURL:URL];
client.username = @"me@email.com";
client.password = @"password";

// setup caching
client.cachePolicy = RKRequestCachePolicyLoadIfOffline | RKRequestCachePolicyLoadOnError | RKRequestCachePolicyTimeout;
client.requestCache.storagePolicy = RKRequestCacheStoragePolicyPermanently;

// setup managed object store
RKObjectManager* objectManager = [RKObjectManager objectManagerWithBaseURL:URL];
RKManagedObjectStore* objectStore = [RKManagedObjectStore   objectStoreWithStoreFilename:@"RestKitCoreDataTest.sqlite"];

// connect my cache implementation
MyCache* cache = [[MyCache alloc] init];
objectStore.managedObjectCache = cache;
objectManager.objectStore = objectStore;

// setup mapping    
RKManagedObjectMapping* userMapping = [RKManagedObjectMapping mappingForClass:[User class]];
[userMapping mapKeyPath:@"id" toAttribute:@"identifier"];
[userMapping mapKeyPath:@"email" toAttribute:@"email"];
[userMapping mapKeyPath:@"firstname" toAttribute:@"firstname"];
[userMapping mapKeyPath:@"surname" toAttribute:@"surname"];
userMapping.primaryKeyAttribute = @"identifier";
[objectManager.mappingProvider setMapping:userMapping forKeyPath:@""];

// setup routes
RKObjectRouter* router = [objectManager router];
[router routeClass:[User class] toResourcePath:@"/users/:identifier"];

User 对象根据 CoreData 支持的需要实现:

@interface User : NSManagedObject

@property (nonatomic, retain) NSNumber * identifier;
@property (nonatomic, retain) NSString * email;
@property (nonatomic, retain) NSString * firstname;
@property (nonatomic, retain) NSString * surname;

@end

@implementation User

@dynamic identifier;
@dynamic email;
@dynamic firstname;
@dynamic surname;

@end

这里是 MyCache。请注意,我不会费心检查resourcePath,因为这只是为了尝试一下,反正我只有一条路。

@implementation MyCache
- (NSArray*)fetchRequestsForResourcePath:(NSString*)resourcePath
{
    NSFetchRequest* fetchRequest = [User fetchRequest];
    return [NSArray arrayWithObject:fetchRequest];
}

-(BOOL)shouldDeleteOrphanedObject:(NSManagedObject *)managedObject
{
    return true;
}
@end

然后我调用服务器以获取路径为“/api/users/123”的 id 为 123 的用户:

User* user = [User object];
user.identifier = [NSNumber numberWithInt:123];
RKObjectManager* manager = [RKObjectManager sharedManager];
[manager getObject:user delegate:self];

这很好用。但是,当我在我的 Mac 上断开 wifi 时,上面的代码不会从 sqlite 数据库中检索用户。

我在委托的objectLoader:didFailWithError 中收到以下错误:

2012-03-01 11:44:09.402 RestKitCoreDataTest[1989:fb03] error: Error Domain=NSURLErrorDomain Code=-1001 "The request timed out." UserInfo=0x6b89aa0 {NSErrorFailingURLStringKey=http://10.211.55.5:3000/api/users/123, NSErrorFailingURLKey=http://10.211.55.5:3000/api/users/123, NSLocalizedDescription=The request timed out., NSUnderlyingError=0x6b81ac0 "The request timed out."}

我认为通过指定在超时时应使用缓存,使用:“RKRequestCachePolicyTimeout”,我希望从本地缓存中检索用户。

缓存确实包含 ID 为 123 的用户记录——在 ZUSER 表中,在 ZIDENTIFIER 列中为 123。

我是否缺少一个步骤来使其正常工作?也许另一个委托方法 需要处理,或者被调用,当缓存命中后 超时?还是我正在尝试做一些你不一定会用 RestKit “开箱即用”的事情?

干杯。

【问题讨论】:

  • 嗨,我正在尝试对 RestKit 0.20.1 做同样的事情——这有什么更新吗?你得到这个工作了吗?您检查的答案似乎没有回答您的问题...谢谢!
  • 我检查了下面的答案是否正确,因为我接受了 Julian 的建议,即使用可达性类来检查我是在线还是离线。后来我最终使用 RestKit 进行在线通信,并自己使用 CoreData 进行离线存储。
  • 好的,谢谢!我想我会尝试将其集中在 appdelegate 中——尽管为了做到这一点,我认为最好创建一个 HTTP 操作队列,然后在每次需要同步时(可能定期)解析它。

标签: objective-c ios restkit


【解决方案1】:

您可以使用可达性类来确定您的客户端是否离线。我经常在每个需要互联网连接的项目中使用这个很棒的课程。

您只需将通知程序启动到特定主机。在你所有的 viewController 中,你现在只需要向 NSNotificationCenter 注册方法来设置一个 BOOL isOnline

通过这种练习,您可以在您的应用中做一些漂亮的事情,例如用流畅的“离线”消息覆盖应用。

https://gist.github.com/1182373

编辑

这是我的一个项目的登录屏幕中的一个示例(抱歉,代码量如此之多,但这是我的完整实现):

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    //// Some stuff ////
    [[Reachability reachabilityWithHostname:@"apple.com"] startNotifier];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(reachabilityChanged:) name:kReachabilityChangedNotification object:nil];
}
- (void)reachabilityChanged:(NSNotification *)note
{
    if ([[note object] isReachable]) {
        CAKeyframeAnimation *scale = [CAKeyframeAnimation animationWithKeyPath:@"transform.scale"];
        [scale setValues:[NSArray arrayWithObjects:[NSNumber numberWithFloat:1.0], [NSNumber numberWithFloat:0.0], nil]];
        [scale setDuration:0.3];
        [scale setRemovedOnCompletion:NO];

        [[offlineView layer] setTransform:CATransform3DMakeScale(0, 0, 1.0)];
        [[offlineView layer] addAnimation:scale forKey:@"scale"];
        [[offlineView layer] setTransform:CATransform3DIdentity];

        [UIView animateWithDuration:0.3 animations:^{
            [offlineView setAlpha:0];
        } completion:^(BOOL finished){
            if (finished) {
                [offlineView removeFromSuperview];
            }
        }];

        [[UIApplication sharedApplication] setStatusBarStyle:UIStatusBarStyleDefault animated:YES];
    }
    else {
        CGRect screenFrame = CGRectMake(0, 0, 320, 480);

        offlineView = [[UIView alloc] initWithFrame:screenFrame];
        [offlineView setBackgroundColor:[UIColor colorWithWhite:0 alpha:0.7]];
        [offlineView setAlpha:0];

        offlineLabel = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 200, 30)];
        [offlineLabel setFont:[UIFont fontWithName:@"Verdana" size:30.0]];
        [offlineLabel setBackgroundColor:[UIColor clearColor]];
        [offlineLabel setTextAlignment:UITextAlignmentCenter];
        [offlineLabel setTextColor:[UIColor whiteColor]];
        [offlineLabel setCenter:[offlineView center]];
        [offlineLabel setText:@"OFFLINE"];

        [offlineView addSubview:offlineLabel];
        [[self window] addSubview:offlineView];

        [UIView animateWithDuration:0.3 animations:^{
            [offlineView setAlpha:1.0];
        }];

        [[UIApplication sharedApplication] setStatusBarStyle:UIStatusBarStyleBlackTranslucent animated:YES];
    }
}

【讨论】:

  • 我使用了这种方法,但我警告你一件事。通知中心将以“随机”顺序发送通知。因此,一个 ViewController 会得到它,而另一个会在稍后得到它,从而导致您的应用程序不一致。也许你会在两个通知之间运行:在 ViewController 上看到“在线”,另一个看到“离线”......我建议只注册一个类,也许是你的 AppDelegate,然后整个应用程序通过一个内部 API 检查这个类。您现在将拥有所有班级的一致视图。
  • @malaba - 你完全正确!我也在管理 AppDelegate 的可达性
  • 你可以注册 RKReachabilityDidChangeNotification 并使用 RKReachabilityObserver 类来使用 RestKit 的东西。
【解决方案2】:

请检查RestKit的版本;在较新版本的 RestKit 中,从托管对象存储中获取缓存数据的方法略有改变。 我没有这台机器上的例子;但如果您需要更多帮助(因为这是一个相当古老的问题),请回复。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2019-03-12
    • 1970-01-01
    • 2015-06-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-01-31
    相关资源
    最近更新 更多