对 RestKit iOS 应用程序的在线和离线支持

Posted

技术标签:

【中文标题】对 RestKit iOS 应用程序的在线和离线支持【英文标题】:Online and offline support for RestKit iOS applications 【发布时间】: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 操作队列,然后在每次需要同步时(可能定期)解析它。 【参考方案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 中,从托管对象存储中获取缓存数据的方法略有改变。 我没有这台机器上的例子;但如果您需要更多帮助(因为这是一个相当古老的问题),请回复。

【讨论】:

以上是关于对 RestKit iOS 应用程序的在线和离线支持的主要内容,如果未能解决你的问题,请参考以下文章

如何在线保存/加载数据(使用 AJAX 和 JSON 存储数据)和离线(本地)

如何在android应用程序中以在线和离线模式加载或检索网页?

实现在线和离线工作的应用程序的最佳方法是啥?

维护网站的在线 (Azure) 和离线版本

phonegap hypride 在线和离线应用程序

关于求解区间第k大的在线和离线做法