我正在构建一个演示程序,并尽可能地符合 ReactiveCocoa design pattern.以下是应用程式的功能:

>查找设备的位置
>每当位置键更改时,提取:

>当前天气
>小时预测
>每日预测

所以顺序是1)更新位置2)合并所有3个天气提取.我建立了一个WeatherManager单例,暴露了天气对象,位置信息和手动更新的方法.此单例符合CLLocationManagerDelegate协议.位置代码是非常基本的,所以我要离开它.唯一真正的兴趣点在于:

- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations {
    // omitting accuracy & cache checking
    CLLocation *location = [locations lastObject];
    self.currentLocation = location;
    [self.locationManager stopUpdatingLocation];
}

获取天气条件都非常相似,所以我已经建立了一种方法来生成一个RACSignal,用于从URL中获取JSON.

- (RACSignal *)fetchJSONFromURL:(NSURL *)url {
    return [RACSignal createSignal:^RACdisposable *(id<RACSubscriber> subscriber) {
        NSURLSessionDataTask *dataTask = [self.session dataTaskWithURL:url completionHandler:^(NSData *data,NSURLResponse *response,NSError *error) {
            if (! error) {
                NSError *jsonError = nil;
                id json = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:&jsonError];
                if (! jsonError) {
                    [subscriber sendNext:json];
                }
                else {
                    [subscriber sendError:jsonError];
                }
            }
            else {
                [subscriber sendError:error];
            }

            [subscriber sendCompleted];
        }];

        [dataTask resume];

        return [RACdisposable disposableWithBlock:^{
            [dataTask cancel];
        }];
    }];
}

这有助于我保持我的方法很好,干净,所以现在我有3个简单的方法构建一个URL并返回RACSignal.这里的好处是我可以创建副作用来解析JSON并分配相应的属性(注意:我在这里使用Mantle).

- (RACSignal *)fetchCurrentConditions {
    // build URL
    return [[self fetchJSONFromURL:url] doNext:^(NSDictionary *json) {
        // simply converts JSON to a Mantle object
        self.currentCondition = [MTLJSONAdapter modelOfClass:[CurrentCondition class] fromJSONDictionary:json error:nil];
    }];
}

- (RACSignal *)fetchHourlyForecast {
    // build URL
    return [[self fetchJSONFromURL:url] doNext:^(NSDictionary *json) {
        // more work
    }];
}

- (RACSignal *)fetchDailyForecast {
    // build URL
    return [[self fetchJSONFromURL:url] doNext:^(NSDictionary *json) {
        // more work
    }];
}

最后,在我的单身人士中,我设置了RAC观察员的位置,因为每次位置改变,我想要获取和更新天气.

[[RACObserve(self,currentLocation)
 filter:^BOOL(CLLocation *newLocation) {
     return newLocation != nil;
 }] subscribeNext:^(CLLocation *newLocation) {
     [[RACSignal merge:@[[self fetchCurrentConditions],[self fetchDailyForecast],[self fetchHourlyForecast]]] subscribeError:^(NSError *error) {
         NSLog(@"%@",error.localizedDescription);
     }];
 }];

一切工作都很好,但是我担心我会从反动的方式偏离构造我的提取和财产分配.我试着用-then做排序:但是没有真正能够得到这个设置我想要的.

我也试图找到一个干净的方式来将异步抓取的结果绑定到单身的属性,但遇到麻烦让它工作.我无法弄清如何“延伸”获取RACSignals(注意:这就是-doNext:想法来自于每一个).

任何帮助清除或资源将是非常好的.谢谢!

解决方法

对于-fetch方法来说,这似乎是不合适的,这样会产生有意义的副作用,这使得我认为你的WeatherManager类混合了两种不同的东西:

>获取最新数据的网络请求
>有状态地存储和呈现数据

这是重要的,因为第一个问题是无国籍的,而第二个几乎完全是有状态的.例如,在GitHub for Mac中,我们使用OCTClient执行联网,然后将返回的用户数据存储在“持久状态管理器”单例中.

一旦你像这样打破它,我认为会更容易理解.您的州长可以与网络客户端进行交互以启动请求,然后州长可以订阅这些请求并应用副作用.

首先,我们让-fetch …方法无状态,通过重写它们来使用转换而不是副作用:

- (RACSignal *)fetchCurrentConditions {
    // build URL
    return [[self fetchJSONFromURL:url] map:^(NSDictionary *json) {
        return [MTLJSONAdapter modelOfClass:[CurrentCondition class] fromJSONDictionary:json error:nil];
    }];
}

然后,您可以使用这些无状态方法并在其中更适合地注入副作用:

- (RACSignal *)updateCurrentConditions {
    return [[self.networkClient
        // If this signal sends its result on a background thread,make sure
        // `currentCondition` is thread-safe,or make sure to deliver it to
        // a kNown thread.
        fetchCurrentConditions]
        doNext:^(CurrentCondition *condition) {
            self.currentCondition = condition;
        }];
}

而且,要更新所有这些,您可以使用merge:(如在您的示例中)与-flattenMap相结合:从位置值映射到新的工作信号:

[[[RACObserve(self,currentLocation)
    ignore:nil]
    flattenMap:^(CLLocation *newLocation) {
        return [RACSignal merge:@[
            [self updateCurrentConditions],[self updateDailyForecast],[self updateHourlyForecast],]];
    }]
    subscribeError:^(NSError *error) {
        NSLog(@"%@",error);
    }];

或者,要在currentLocation发生更改时自动取消飞行中更新,请将-flattenMap替换为-switchToLatest:

[[[[RACObserve(self,currentLocation)
    ignore:nil]
    map:^(CLLocation *newLocation) {
        return [RACSignal merge:@[
            [self updateCurrentConditions],]];
    }]
    switchToLatest]
    subscribeError:^(NSError *error) {
        NSLog(@"%@",error);
    }];

(ReactiveCocoa/ReactiveCocoa#786原始答复).

ios – 具有异步网络请求的ReactiveCocoa排序的更多相关文章

  1. ios – 使用ReactiveCocoa时未调用错误块

    出于某种原因,我没有收到错误消息.这打印“B”.知道为什么吗?如果在订阅者上调用-sendError:,为什么完成块会收到它?

  2. ios – RACObserve不工作

    我正在尝试观察我的viewmodel中的属性,然后使用ReactiveCocoa更新带有它的值的标签,但它没有更新.这是我得到的:视图模型视图控制器我检查过,viewmodel正在更新’amount’属性.这里有什么我想念的吗?我也试过这个测试:也不起作用.我正在使用ReactiveCocoa2.4.7,因为我的应用程序支持iOS7.Swift[1,2]和此版本中的宏替换之间是否存在任何不兼容性?

  3. ios – 为什么在ReactiveCocoa中调用信号两次?

    我用https://github.com/ReactiveCocoa/ReactiveCocoa实现了我的第一个代码.用于登录用户.行[subscribersendNext:user];被叫两次,但我希望只有一次.并且根本没有调用地图(因此永远不会调用自动登录)这是我的实施:这样称呼:解决方法发生这种情况是因为每当对信号进行订阅时,传递给[RACSignalcreateSignal:]的块都会执行

  4. ios – AFNetworking / NSURLConnection接收NSPOSIXErrorDomain代码= 9“操作无法完成.坏文件描述符“

    有人在他们的AFNetworking操作中遇到这个错误吗?此外,如果我真的想要,如何故意关闭这个文件描述符?

  5. ios – 具有异步网络请求的ReactiveCocoa排序

    我正在构建一个演示程序,并尽可能地符合ReactiveCocoadesignpattern.以下是应用程式的功能:>查找设备的位置>每当位置键更改时,提取:>当前天气>小时预测>每日预测所以顺序是1)更新位置2)合并所有3个天气提取.我建立了一个WeatherManager单例,暴露了天气对象,位置信息和手动更新的方法.此单例符合CLLocationManagerDelegate协议.位置代码是非

  6. ios – 使用ReactiveCocoa跟踪远程对象的UI更新

    我正在制作一个iOS应用程序,可让您远程控制桌面上播放的应用程序中的音乐.最难的问题之一是能够正确地更新“跟踪器”的位置(其显示当前正在播放的歌曲的时间位置和持续时间).这里有几个输入源:>启动时,遥控器发送网络请求以获取当前播放歌曲的初始位置和持续时间.>当用户使用遥控器调整跟踪器的位置时,它向音乐应用发送网络请求以改变歌曲的位置.>如果用户使用桌面上的应用程序更改跟踪器的位置,则应用程序会使用

  7. ios – ReactiveCocoa rac_valuesForKeyPath在Swift中不起作用

    解决方法可观察性需要是动态的我让您的示例使用以下代码

  8. swift语言IOS8开发战记24 解析Json

    在IOS中使用NSURLConnection实现http通信,NSURLConnection提供了异步和同步两种通信方式,同步请求会造成进程阻塞,通常我们使用异步的方式,不管同步还是异步,建立通信的基本步骤都是一样的:1,创建NSURL2,创建Request对象3,创建NSURLConnection连接第3步结束后就建立了一个http连接。

  9. swift网络数据请求方法

    在我们的工程文件里面饮入Alamofire便可以使用它了通过Alamofire可以只需要几行代码就完成网络数据的请求,上面的的parameters是给PHP发了一个post请求,请求名称是name,值为“jimmy”,这样label上就会快速的显示返回的json数据,由于alamofire本就是异步请求,所以不必像第一种方法那样在更新UI的时候,还要跳到主线程,在Alamofire中,请求只要返回正确,便可以在必包函数中任意的做UI上的操作了

  10. Swift UITableView瀑布流/NSURLConnection异步网络请求

    去年写过一个OC版本的瀑布流Demo《UITableView实现的瀑布流效果》。接触Swift一段时间了,今天就是用Swift再写了一个瀑布流的Demo。原理是一样的这里不再赘述。在写这个Demo的过程中是用到了NSURLConnection的异步网络请求和GCD做了一个图片的异步加载,没有做图片的缓存,所以是用起来有些卡。cell是带左滑删除视图的,删除功能是没有实现的。O(∩_∩)O哈哈~

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部