- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我编写了一个非常简单的 ReactiveCocoa 测试应用程序来尝试在 RAC 中编码(而不是无休止地阅读它)。在 Github 上,我想得到一些关于它的具体问题的答案。我将链接到代码组件。
首先,对该应用程序进行简要说明:它是一个定时器驱动的迭代计数器,可以由用户暂停。 (它的目的是计算已经过去了多少秒,省略了用户暂停它的秒数。)每秒一次,如果用户没有暂停递增行为,计时器递增一个变量。
我关心三个类的听觉反馈:
BOOL
属性来控制是否运行累加。MPSTicker
到 View Controller 的 ViewModel 包装。它提供只读字符串,显示滴答数并显示操作文本,如果采取该操作,“暂停”或“恢复”滴答。它还有一个可读写的 BOOL
用于暂停/取消暂停报价。MPSViewModel
中的字符串.我的问题:
BOOL
property on MPSTicker
用于启用/禁用它的积累,但我不知道如何更被动地做到这一点。 (这也在 ViewModel 和 ViewController 的下游运行:我如何通过所有这三个运行一个字符串来控制代码是否正在运行?)tickString
和 tickStateString
作为非常传统的属性,但是使用这些属性的 ViewController 会立即将它们映射回 text on a label和 button text使用 RACObserve
。这感觉不对,但我不知道如何公开来自 ViewModel 的信号,以便 ViewController 轻松使用这两个属性。paused
BOOL
时。我认为这是 #1 的另一个下游效应,“这不应该是 BOOL
属性”,但我不确定(注意:我想我回避了 MPSTicker
上 paused
的 BOOL
信号,因为我不知道如何在 ViewModel 中使用它来派生两个字符串(一个用于当前滴答计数,一个用于操作文本),也不是如何在用户按下“暂停”或“恢复”按钮时推送 UI 驱动的值更改。这是我在问题 1 和 3 中的核心关注点。)
一些屏幕截图可以帮助您形象化这个华丽的设计:
滴答作响:
暂停:
最佳答案
这是一篇很棒的文章!
I don't like the BOOL property on MPSTicker for enabling/disabling its accumulation, but I didn't know how to do it more Reactive-ly. (This also runs downstream to the ViewModel and ViewController: how can I run a string through all three of these to control whether or not the ticker is running?)
从广义上讲,使用属性并没有错或非响应式。 KVO-able 属性可以被认为是学术 FRP 意义上的行为:它们是在其生命周期的所有时间点都具有值(value)的信号。事实上,在 Objective-C 中,属性甚至比信号更好,因为它们保留了类型信息,否则我们会通过将其包装在 RACSignal
中而丢失这些信息。
因此,如果它是完成这项工作的正确工具,那么使用支持 KVO 的属性并没有错。只要倾斜你的头,稍微眯一下,它们看起来就像信号。
某物应该是属性还是 RACSignal
更多是关于您试图捕获的语义。您需要一个属性的特性(哈哈!),还是您更关心值随时间变化的一般概念?
在 MPSTicker
的特定情况下,我认为 accumulateEnabled
的转换才是您真正关心的事情。
所以如果 MPSTicker
有一个 accumulationEnabledSignal
属性,我们会做类似的事情:
_accumulateSignal = [[[[RACSignal
combineLatest:@[ _tickSignal, self.accumulationEnabledSignal ]]
filter:^(RACTuple *t) {
NSNumber *enabled = t[1];
return enabled.boolValue;
}]
reduceEach:^(NSNumber *tick, NSNumber *enabled) {
return tick;
}]
scanWithStart:@(0) reduce:^id(NSNumber *previous, id next) {
// On each tick, we add one to the previous value of the accumulate signal.
return @(previous.unsignedIntegerValue + 1);
}];
我们将 tick 和 enabledness 结合起来,因为正是这两者的转换驱动了我们的逻辑。
(FWIW,RACCommand
类似并使用启用信号:https://github.com/ReactiveCocoa/ReactiveCocoa/blob/9503c6ef7f2f327f4db6440ddfbc4ee09b86857f/ReactiveCocoaFramework/ReactiveCocoa/RACCommand.h#L95。)
The ViewModel exposes tickString and tickStateString as very traditional properties, but the ViewController which consumes these immediately maps them back into text on a label and button text with RACObserve. This feels wrong, but I don't know how to expose a signal from the ViewModel that's easy for the ViewController to consume for these two attributes.
我可能没有理解您的意思,但我认为您的描述很好。这又回到了上面关于属性和信号之间关系的观点。
使用 RAC 和 MVVM,很多代码只是将数据线程化到应用程序的其他部分,根据特定上下文的需要对其进行转换。它是关于通过应用程序的数据流。这很无聊——几乎是机械的——但这就是重点。我们必须以特定方式重新发明或处理的越少越好。
FWIW,我会稍微更改一下实现:
RAC(self, tickString) = [[[[_ticker
accumulateSignal]
deliverOn:[RACScheduler mainThreadScheduler]]
// Start with 0.
startWith:@(0)]
map:^(NSNumber *tick) {
// Unpack the value and format our string for the UI.
NSUInteger count = tick.unsignedIntegerValue;
return [NSString stringWithFormat:@"%i tick%@ since launch", count, (count != 1 ? @"s" : @"")];
}];
这样我们就可以更明确地定义 tickString
与 ticker
的某种转换之间的关系(并且我们可以避免执行强/弱 self
跳舞)。
The ViewController suffers an indignity when flipping the paused BOOL on the ViewModel. I think this is another downstream effect of #1, "This shouldn't be a BOOL property", but I'm not sure
我可能只是因为疲倦而想念它,但你在这里想到的侮辱是什么?
关于ios - 在我的 ReactiveCocoa 测试项目中了解 ReactiveCocoa 和 MVVM,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21419864/
我正在使用 NetBeans 开发 Java 中的 WebService,并使用 gradle 作为依赖管理。 我找到了this article关于使用 gradle 开发 Web 项目。它使用 Gr
我正在将旧项目从 ant 迁移到 gradle(以使用其依赖项管理和构建功能),并且在生成 时遇到问题>eclipse 项目。今天的大问题是因为该项目有一些子项目被拆分成 war 和 jar 包部署到
我已经为这个错误苦苦挣扎了很长时间。如果有帮助的话,我会提供一些问题的快照。请指导我该怎么办????在我看来,它看起来一团糟。 *** glibc detected *** /home/shivam/
我在 Ubuntu 12.10 上运行 NetBeans 7.3。我正在学习 Java Web 开发类(class),因此我有一个名为 jsage8 的项目,其中包含我为该类(class)所做的工作。
我想知道 Codeplex、GitHub 等中是否有任何突出的项目是 C# 和 ASP.NET,甚至只是 C# API 与功能测试 (NUnit) 和模拟(RhinoMocks、NMock 等)。 重
我创建了一个 Maven 项目,包装类型为“jar”,名为“Y”我已经完成了“Maven 安装”,并且可以在我的本地存储库中找到它.. 然后,我创建了另一个项目,包装类型为“war”,称为“X”。在这
我一直在关注the instructions用于将 facebook SDK 集成到我的应用程序中。除了“helloFacebookSample”之外,我已经成功地编译并运行了所有给定的示例应用程序。
我想知道,为什么我们(Java 社区)需要 Apache Harmony 项目,而已经有了 OpenJDK 项目。两者不是都是在开源许可下发布的吗? 最佳答案 事实恰恰相反。 Harmony 的成立是
我正在尝试使用 Jsoup HTML Parser 从网站获取缩略图 URL我需要提取所有以 60x60.jpg(或 png)结尾的 URL(所有缩略图 URL 都以此 URL 结尾) 问题是我让它在
我无法构建 gradle 项目,即使我编辑 gradle 属性,我也会收到以下错误: Error:(22, 1) A problem occurred evaluating root project
我有这个代码: var NToDel:NSArray = [] var addInNToDelArray = "Test1 \ Test2" 如何在 NToDel:NSArray 中添加 addInN
如何在单击显示更多(按钮)后将主题列表限制为 5 个(项目)。 还有 3(项目),依此类推到列表末尾,然后它会显示显示更少(按钮)。 例如:在 Udemy 过滤器选项中,当您点击查看更多按钮时,它仅显
如何将现有的 Flutter 项目导入为 gradle 项目? “导入项目”向导要求 Gradle 主路径。 我有 gradle,安装在我的系统中。但是这里需要设置什么(哪条路径)。 这是我正在尝试的
我有一个关于 Bitbucket 的项目。只有源被提交。为了将项目检索到新机器上,我在 IntelliJ 中使用了 Version Control > Checkout from Ve
所以,我想更改我公司的一个项目,以使用一些与 IDE 无关的设置。我在使用 Tomcat 设置 Java 应用程序方面有非常少的经验(我几乎不记得它是如何工作的)。 因此,为了帮助制作独立于 IDE
我有 2 个独立的项目,一个在 Cocos2dx v3.6 中,一个在 Swift 中。我想从 Swift 项目开始游戏。我该怎么做? 我已经将整个 cocos2dx 项目复制到我的 Swift 项目
Cordova 绝对是新手。这些是我完成的步骤: checkout 现有项目 运行cordova build ios 以上生成此构建错误: (node:10242) UnhandledPromiseR
我正在使用 JQuery 隐藏/显示 li。我的要求是,当我点击任何 li 时,它应该显示但隐藏所有其他 li 项目。当我将鼠标悬停在文本上时 'show all list item but don
我想将我所有的java 项目(223 个项目)迁移到gradle 项目。我正在使用由 SpringSource STS 团队开发的 Gradle Eclipse 插件。 目前,我所有的 java 项目
我下载this Eclipse Luna ,对于 Java EE 开发人员,如描述中所见,它支持 Web 应用程序。我找不到 file -> new -> other -> web projects
我是一名优秀的程序员,十分优秀!