- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个用于保存数据元素数组的 UITableView 的 ViewModel。 UITableView 实现了下拉刷新和无限滚动行为。数据元素是通过 RestKit 从服务器以分页方式请求的,这意味着我必须以某种方式跟踪当前页面。我创建了 2 个单独的 RACCommands 来区分刷新和无限滚动加载,其中刷新总是加载第 0 页。整个过程有效,但我不喜欢它,想知道是否有更好的方法来做到这一点。另外,现在这 2 个命令可能会同时执行,这不是预期的。
@interface TableDataViewModel ()
@property(nonatomic, readonly) RestApiConnector *rest;
@property(nonatomic) int page;
@property(nonatomic) NSMutableArray *data;
@property(nonatomic) RACCommand *loadCommand;
@property(nonatomic) RACCommand *reloadCommand;
@end
@implementation TableDataViewModel
objection_requires_sel(@selector(rest))
- (id)init {
self = [super init];
if (self) {
[self configureLoadCommands];
[self configureActiveSignal];
}
return self;
}
- (void)configureActiveSignal {
[self.didBecomeActiveSignal subscribeNext:^(id x) {
if (!self.data) {
[self.reloadCommand execute:self];
}
}];
}
- (void)configureLoadCommands {
self.loadCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
return [self.rest dataSignalWithPage:self.page];
}];
self.reloadCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
return [self.rest dataSignalWithPage:self.page];
}];
self.loadCommand.allowsConcurrentExecution = FALSE;
self.reloadCommand.allowsConcurrentExecution = FALSE;
[self.reloadCommand.executionSignals subscribeNext:^(id x) {
RAC(self, data) = [x map:^id(id value) {
self.page = 0;
return [value mutableCopy];
}];
}];
[self.loadCommand.executionSignals subscribeNext:^(id x) {
RAC(self, data) = [x map:^id(id value) {
self.page++;
if (self.data) {
NSMutableArray *array = [self.data mutableCopy];
[array addObjectsFromArray:value];
return array;
} else {
return [value mutableCopy];
}
}];
}];
}
@end
最佳答案
I just started using ReactiveCocoa, so I'd appreciate any other tips for this.
1 const NSUInteger kLoad = 0;
2 const NSUInteger kReload = 1;
3
4 - (void)configureActiveSignal {
5 @weakify(self);
6 RACSignal *dba = [[self.didBecomeActiveSignal filter:^(id _) {
7 @strongify(self);
8 return self.data;
9 }]
10 mapReplace:@( kReload )];
11
12 RACSignal *ls = [self.command rac_liftSelector:@selector(execute:) withSignals:dba];
13 [[ls publish] connect];
14 }
15
16 - (void)configureCommand {
17 @weakify(self);
18 self.command = [[RACCommand alloc] initWithSignalBlock:^(NSNumber *loadOrReload) {
19 @strongify(self);
20 return RACTuplePack(loadOrReload, [self.rest dataSignalWithPage:self.page]);
21 }];
22
23 RAC(self, page) = [self.command.executionSignals reduceEach:^(NSNumber *loadOrReload, id _) {
24 @strongify(self);
25 return kReload == loadOrReload.integerValue ? @0 : @( ++self.page );
26 }];
27 RAC(self, data) = self.command.executionSignals reduceEach:^(NSNumber *loadOrReload, NSArray *data) {
28 @strongify(self);
29 if (kReload == loadOrReload.integerValue)
30 {
31 return [data mutableCopy];
32 }
33 else
34 {
35 NSMutableArray *ma = [self.data mutableCopy];
36 [ma addObjectsFromArray:data]
37 return ma;
38 }
39 }];
40
41 self.command.allowsConcurrentExecution = NO;
command
。现在不再使用数据数组调用它,而是使用一个常量调用它来指示它是应该加载还是重新加载,您可以在第 18 行看到。 self
强烈引用的块中(直接或通过对象所有权的间接链)强烈引用 self
。 ReactiveCocoa 附带了 @strongify/weakify
宏来帮助减少这个难度。如您所见,任何时候您需要在命令拥有的块中引用 self
(它本身由 self
拥有,因此循环),您需要创建对 self 的弱引用,然后在块内使用它.您在块内使用 @strongify
以避免与 ARC 的释放机制发生数据竞争。 didBecomeActive
参数将 self.command
通知转换为 kReload
的执行(这会替换旧的 reloadCommand
),但前提是 self.data != nil
,使用 -filter:
。 [self.command execute:]
的地方,但您可以通过将选择器提升到上面刚刚创建的信号上来实现。大概你有一些其他未显示的代码用于执行 self.loadCommand
,你需要调整它以传递 @( kLoad )
。 -publish/connect
习惯用法通常用于当您想订阅信号而不实际对其值进行任何操作时。这在 RAC 2.x 中并不是绝对必要的。 -execute:
输入和数据的元组。这是因为正如您将在下面看到的,下游操作需要访问这两条信息。 RAC()
宏,而是直接在 -configureCommand
方法中使用它。这里我们说的是,无论何时执行命令,都将 self.page
设置为 0
或 ++self.page
,具体取决于命令是使用 kLoad
还是 kReload
执行的。 -reduceEach:
操作只是为了方便解构通过信号发送的元组,这里我们只关心常量。 注意在这个 self.page
操作中增加 -reduceEach:
的副作用。 在信号操作中产生副作用通常是不好的形式。危险在于,如果 RAC(self, page)
以外的其他东西订阅了返回的信号,则实例变量的递增次数可能会超出您的预期。如果您能以某种方式从数据源获取 self.page
的值,而不是在全局实例变量中跟踪它,那将是理想的。 self.page
,而是设置 self.data
。这次 -reduceEach:
操作确实需要使用元组中的第二个值(数据),该值被适本地转换为 kLoad
或 kReload
,然后最终设置在 ivar 上。 关于reactive-cocoa - 如何使用 RACCommands 处理表数据请求,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22539255/
对于 Metal ,如果对主纹理进行 mipmap 处理,是否还需要对多采样纹理进行 mipmap 处理?我阅读了苹果文档,但没有得到任何相关信息。 最佳答案 Mipmapping 适用于您将从中
我正在使用的代码在后端 Groovy 代码中具有呈现 GSP(Groovy 服务器页面)的 Controller 。对于前端,我们使用 React-router v4 来处理路由。我遇到的问题是,通过
我们正在 build 一个巨大的网站。我们正在考虑是在服务器端(ASP .Net)还是在客户端进行 HTML 处理。 例如,我们有 HTML 文件,其作用类似于用于生成选项卡的模板。服务器端获取 HT
我正在尝试将图像加载到 void setup() 中的数组中,但是当我这样做时出现此错误:“类型不匹配,'processing .core.PImage' does not匹配“processing.
我正在尝试使用其私有(private)应用程序更新 Shopify 上的客户标签。我用 postman 尝试过,一切正常,但通过 AJAX,它带我成功回调而不是错误,但成功后我得到了身份验证链接,而不
如何更改我的 Processing appIconTest.exe 导出的默认图标在窗口中的应用程序? 默认一个: 最佳答案 经过一些研究,我能找到的最简单的解决方案是: 进入 ...\process
我在 Processing 中做了一个简单的小游戏,但需要一些帮助。我有一个 mp3,想将它添加到我的应用程序中,以便在后台循环运行。 这可能吗?非常感谢。 最佳答案 您可以使用声音库。处理已经自带
我有几个这样创建的按钮: 在 setup() PImage[] imgs1 = {loadImage("AREA1_1.png"),loadImage("AREA1_2.png"),loadImage
我正在尝试使用 Processing 创建一个多人游戏,但无法弄清楚如何将屏幕分成两个以显示玩家的不同情况? 就像在 c# 中一样,我们有Viewport leftViewport,rightView
我一直在尝试使用 Moore 邻域在处理过程中创建元胞自动机,到目前为止非常成功。我已经设法使基本系统正常工作,现在我希望通过添加不同的功能来使用它。现在,我检查细胞是否存活。如果是,我使用 fill
有没有办法用 JavaScript 代码检查资源使用情况?我可以检查脚本的 RAM 使用情况和 CPU 使用情况吗? 由于做某事有多种方法,我可能会使用不同的方法编写代码,并将其保存为两个不同的文件,
我想弄清楚如何处理这样的列表: [ [[4,6,7], [1,2,4,6]] , [[10,4,2,4], [1]] ] 这是一个整数列表的列表 我希望我的函数将此列表作为输入并返回列表中没有重复的整
有没有办法在不需要时处理 MethodChannel/EventChannel ?我问是因为我想为对象创建多个方法/事件 channel 。 例子: class Call { ... fields
我有一个关于在 Python3 中处理 ConnectionResetError 的问题。这通常发生在我使用 urllib.request.Request 函数时。我想知道如果我们遇到这样的错误是否可
我一直在努力解决这个问题几个小时,但无济于事。代码很简单,一个弹跳球(粒子)。将粒子的速度初始化为 (0, 0) 将使其保持上下弹跳。将粒子的初始化速度更改为 (0, 0.01) 或任何十进制浮点数都
我把自己弄得一团糟。 我想在我的系统中添加 python3.6 所以我决定在我的 Ubuntu 19.10 中卸载现有的。但是现在每次我想安装一些东西我都会得到这样的错误: dpkg: error w
我正在努力解决 Rpart 包中的 NA 功能。我得到了以下数据框(下面的代码) Outcome VarA VarB 1 1 1 0 2 1 1 1
我将 Java 与 JSF 一起使用,这是 Glassfish 3 容器。 在我的 Web 应用程序中,我试图实现一个文件(图像)管理系统。 我有一个 config.properties我从中读取上传
所以我一直在Processing工作几个星期以来,虽然我没有编程经验,但我已经转向更复杂的项目。我正在编写一个进化模拟器,它会产生具有随机属性的生物。 最终,我将添加复制,但现在这些生物只是在屏幕上漂
有人知道 Delphi 2009 对“with”的处理有什么不同吗? 我昨天解决了一个问题,只是将“with”解构为完整引用,如“with Datamodule、Dataset、MainForm”。
我是一名优秀的程序员,十分优秀!