- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试学习 RxJS observables。在柜台上编写了一个示例应用程序。
用户可以在页面上看到所有计数器的总数。 (不是计数器的数量,而是所有计数器值的总和)。
例如:
页面上有两个计数器,其值分别为 2,3。计数器总数应为 2+3 = 5。
如果计数器 1 被移除,计数器总计应为 5-2 = 3
我遇到的问题是每当我点击删除按钮时,它应该从计数器总数中扣除该计数器值。我在删除按钮上使用了一个主题来观察它并解决了。
后来我添加了 removeAll 按钮,我遇到了与以前类似的问题。我无法将计数器总流设置为 0。
我尝试了 .last() 方法,但由于可观察对象未结束,我无法获得计数器总可观察对象的最后状态。
我使用了 merge() 但它没有解决问题。
我解决不了。
我想重新添加一个主题,但是我遇到了两次这个问题,现在我想知道是否有比添加主题更好的解决方案?或者我可能遗漏了什么。
```
// Code goes here
function createCounter(number){
return "<div class='counter' id='counter" + number + "'>" +
'<button id="increment' + number + '">+</button>' +
'<h1 style="display:inline-block; margin: 10px" id="counterValue' + number + '"></h1>' +
'<button id="decrement' + number + '">-</button>' +
'<button id="remove' + number + '">Remove</button>'
"</div>";
}
$(document).ready(function(){
var addCounter$ = Rx.Observable.fromEvent($("#addCounter"), 'click')
.map(()=> 1)
.startWith(0)
.scan((x,y) => x+y);
var countersSubject = new Rx.Subject();
var removeAll$ = Rx.Observable.fromEvent($("#removeAll"), 'click').map(() => 0).startWith(0);
var countersTotal$ = countersSubject.startWith(0).scan((x,y) => x+y).merge(removeAll$);
removeAll$.subscribe(() =>{
$('#counterContainer').empty();
});
countersTotal$.subscribe(total => {
$('#countersTotal').text(total);
});
addCounter$.subscribe(counterNum => {
$('#counterContainer').append(createCounter(counterNum));
var increment$ = Rx.Observable.fromEvent($("#increment" + counterNum), 'click')
.map(() => +1);
var decrement$ = Rx.Observable.fromEvent($("#decrement" + counterNum), 'click')
.map(() => -1);
var action$ = Rx.Observable.merge(increment$, decrement$);
var state$ = action$.startWith(0).scan((prev, now) => prev+now);
var counterSubs = state$.subscribe(val => {
$("#counterValue" + counterNum).text(val);
});
var countersTotalSubs = action$.subscribe(countersSubject);
var remove$ = Rx.Observable.fromEvent($("#remove" + counterNum), 'click');
var removeCounterTotal$ = Rx.Observable.combineLatest(remove$, state$,
(x,y) => -y);
removeCounterTotal$.subscribe(countersSubject);
remove$.subscribe(remove => {
countersTotalSubs.dispose();
counterSubs.dispose();
$('div').remove('#counter' + counterNum);
});
});
});
请看这里。 http://plnkr.co/flJx4LNRiboPmW0GjZtl
如果您需要什么,请告诉我。
谢谢
最佳答案
你是对的,你可以不用Subjects
。我会说这是一个相当困难的挑战,要立即开始(所以支持你!),但肯定是可行的。
您已经很好地理解了“一切皆流”,所以让我们分解一下。首先,每个计数器都是它自己的组件,它是几个不同流(递增、递减、删除)的合并。因此,让我们从那里开始,然后从那里向外移动。
首先简化您的流以增加、减少和删除,因为它也是一种行为:
$('#counterContainer').append(createCounter(counterNum));
//The map operator can take a value which it will map to every value it receives
var inc$ = Rx.Observable.fromEvent($('#increment' + counterNum), 'click').map(+1);
var dec$ = Rx.Observable.fromEvent($('#decrement' + counterNum), 'click').map(-1);
var remove$ = Rx.Observable.fromEvent($('#remove' + counterNum), 'click');
接下来我们使用merge
+ scan
技术来保持计数器值的运行总和。
return Rx.Observable.merge(inc$, dec$)
.startWith(0)
.scan((prev, now) => prev + now);
但现在我们开始第一个转折,我们知道我们只想获取值直到计数器被移除(注意强调),进一步,我们知道通过单击删除我们实际上想从 div 中删除计数器。通过结合这些想法,我们可以添加两种新行为:
return Rx.Observable.merge(inc$, dec$)
.startWith(0)
.scan((prev, now) => prev + now)
//Complete when the remove button is clicked
.takeUntil(remove$)
//When completed remove this counter
.finally(() => $('div').remove('#counter' + counterNum))
//Show the value
.do(val => $('#counterValue' + counterNum).text(val));
为了让所有计数器能够同时运行,我们应该将它们合并在一起,因为我们实际上需要两个值,一个总值和一个增量值,我们将拆分出两个流,一个将在内部用于更新计数器的值和另一个是增量,用于更新总值。
为此,您可以使用 share
和 shareReplay
以及 using
运算符将所有这些流连接在一起。
//flatMap has an index parameter which can be used here to tally the total
//number of counters "in-flight"
var counters = addCounter$.flatMap((counterNum, idx) => {
$('#counterContainer').append(createCounter(counterNum));
var inc$ = Rx.Observable.fromEvent($('#increment' + counterNum), 'click').map(+1);
var dec$ = Rx.Observable.fromEvent($('#decrement' + counterNum), 'click').map(-1);
//Merges all the events together to describe their logic and then
//shares the resulting Observable
var counter = Rx.Observable.merge(inc$, dec$)
.takeUntil(remove$)
.startWith(0)
.share();
//Creates an Observable that will always emit the last value it
//recieved to all new subscribers
var total = counter
.scan((prev, now) => prev + now, 0)
.shareReplay(1);
return Rx.Observable.using(
//Starts the `total` Observable and updates the counter value
//when a button is pressed
//Ties the subscription's lifetime to that of `counter`
() => total.subscribe(val => $('#counterValue' + counterNum).text(val)),
//Returns the counter Observable
() => counter
)
.finally(() => $('div').remove('#counter' + counterNum))
//When the above Observable completes we will emit one last message
//which will be the total * -1 (subtracting the value from the overall total)
.concat(total.last().map(x => x * -1));
});
现在定义 addCounter$
的地方:
var addCounter$ = Rx.Observable.fromEvent($("#addCounter"), 'click')
//Map also takes an index parameter which can be leveraged here
.map((_, idx) => idx);
最后我们需要将其组合在一起,拼图的最后一 block 是removeAll
功能。到目前为止,我们所看到的所有内容都可以被认为是此功能的子流,因为全部删除有点像“恢复”状态。我们可以获取 counters
流并将其包装在每次点击 remove all
时重新启动的流中,因为我们的内部 Observables 会自动清理,我们也会神奇地删除他们都在这个过程中。
var removeAll$ = Rx.Observable.fromEvent($("#removeAll"), 'click');
removeAll$
.startWith(0)
.flatMapLatest(() => {
resetTotal();
//Total all the deltas from all of the counters
return adder.scan((acc, val) => acc + val);
})
.subscribe(x => $('#countersTotal').text(x));
仅此而已!请参阅上面我更新的 plunkr 和工作示例(也在下面复制)。
// Code goes here
function createCounter(number){
return "<div class='counter' id='counter" + number + "'>" +
'<button id="increment' + number + '">+</button>' +
'<h1 style="display:inline-block; margin: 10px" id="counterValue' + number + '"></h1>' +
'<button id="decrement' + number + '">-</button>' +
'<button id="remove' + number + '">Remove</button>'
"</div>";
}
$(document).ready(function(){
var addCounter$ = Rx.Observable.fromEvent($("#addCounter"), 'click')
.map((_, idx) => idx);
function resetTotal() {
$('#countersTotal').text(0);
}
var removeAll$ = Rx.Observable.fromEvent($("#removeAll"), 'click');
var adder = addCounter$.flatMap((counterNum, idx) => {
$('#counterContainer').append(createCounter(counterNum));
var inc$ = Rx.Observable.fromEvent($('#increment' + counterNum), 'click').map(+1);
var dec$ = Rx.Observable.fromEvent($('#decrement' + counterNum), 'click').map(-1);
var remove$ = Rx.Observable.fromEvent($('#remove' + counterNum), 'click');
var counter = Rx.Observable.merge(inc$, dec$)
.takeUntil(remove$)
.startWith(0)
.share();
var total = counter
.scan((prev, now) => prev + now, 0)
.shareReplay(1);
var d = total
.subscribe(val => $('#counterValue' + counterNum).text(val));
return Rx.Observable.using(() => d, () => counter)
.finally(() => $('div').remove('#counter' + counterNum))
.concat(total.last().map(x => x * -1));
});
removeAll$
.startWith(0)
.flatMapLatest(() => {
resetTotal();
return adder.scan((acc, val) => acc + val);
})
.subscribe(x => $('#countersTotal').text(x));
});
<!DOCTYPE html>
<html>
<head>
<script data-require="jquery@2.2.0" data-semver="2.2.0" src="https://ajax.googleapis.com/ajax/libs/jquery/2.2.0/jquery.min.js"></script>
<link rel="stylesheet" href="style.css" />
<script data-require="rxjs@4.1.0" data-semver="4.1.0" src="//cdnjs.cloudflare.com/ajax/libs/rxjs/4.1.0/rx.all.js"></script>
<script src="script.js"></script>
</head>
<body>
<div id ="app">
<button id="addCounter">Add counter</button>
<button id="removeAll">Remove all</button>
<h1 >Counters Total <span id="countersTotal" ></span></h1>
<div id="counterContainer"></div>
</div>
</body>
</html>
关于javascript - 卡在使用 RxJS 删除计数器应用程序中的所有内容,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37028892/
我在网上搜索但没有找到任何合适的文章解释如何使用 javascript 使用 WCF 服务,尤其是 WebScriptEndpoint。 任何人都可以对此给出任何指导吗? 谢谢 最佳答案 这是一篇关于
我正在编写一个将运行 Linux 命令的 C 程序,例如: cat/etc/passwd | grep 列表 |剪切-c 1-5 我没有任何结果 *这里 parent 等待第一个 child (chi
所以我正在尝试处理文件上传,然后将该文件作为二进制文件存储到数据库中。在我存储它之后,我尝试在给定的 URL 上提供文件。我似乎找不到适合这里的方法。我需要使用数据库,因为我使用 Google 应用引
我正在尝试制作一个宏,将下面的公式添加到单元格中,然后将其拖到整个列中并在 H 列中复制相同的公式 我想在 F 和 H 列中输入公式的数据 Range("F1").formula = "=IF(ISE
问题类似于this one ,但我想使用 OperatorPrecedenceParser 解析带有函数应用程序的表达式在 FParsec . 这是我的 AST: type Expression =
我想通过使用 sequelize 和 node.js 将这个查询更改为代码取决于在哪里 select COUNT(gender) as genderCount from customers where
我正在使用GNU bash,版本5.0.3(1)-发行版(x86_64-pc-linux-gnu),我想知道为什么简单的赋值语句会出现语法错误: #/bin/bash var1=/tmp
这里,为什么我的代码在 IE 中不起作用。我的代码适用于所有浏览器。没有问题。但是当我在 IE 上运行我的项目时,它发现错误。 而且我的 jquery 类和 insertadjacentHTMl 也不
我正在尝试更改标签的innerHTML。我无权访问该表单,因此无法编辑 HTML。标签具有的唯一标识符是“for”属性。 这是输入和标签的结构:
我有一个页面,我可以在其中返回用户帖子,可以使用一些 jquery 代码对这些帖子进行即时评论,在发布新评论后,我在帖子下插入新评论以及删除 按钮。问题是 Delete 按钮在新插入的元素上不起作用,
我有一个大约有 20 列的“管道分隔”文件。我只想使用 sha1sum 散列第一列,它是一个数字,如帐号,并按原样返回其余列。 使用 awk 或 sed 执行此操作的最佳方法是什么? Accounti
我需要将以下内容插入到我的表中...我的用户表有五列 id、用户名、密码、名称、条目。 (我还没有提交任何东西到条目中,我稍后会使用 php 来做)但由于某种原因我不断收到这个错误:#1054 - U
所以我试图有一个输入字段,我可以在其中输入任何字符,但然后将输入的值小写,删除任何非字母数字字符,留下“。”而不是空格。 例如,如果我输入: 地球的 70% 是水,-!*#$^^ & 30% 土地 输
我正在尝试做一些我认为非常简单的事情,但出于某种原因我没有得到想要的结果?我是 javascript 的新手,但对 java 有经验,所以我相信我没有使用某种正确的规则。 这是一个获取输入值、检查选择
我想使用 angularjs 从 mysql 数据库加载数据。 这就是应用程序的工作原理;用户登录,他们的用户名存储在 cookie 中。该用户名显示在主页上 我想获取这个值并通过 angularjs
我正在使用 autoLayout,我想在 UITableViewCell 上放置一个 UIlabel,它应该始终位于单元格的右侧和右侧的中心。 这就是我想要实现的目标 所以在这里你可以看到我正在谈论的
我需要与 MySql 等效的 elasticsearch 查询。我的 sql 查询: SELECT DISTINCT t.product_id AS id FROM tbl_sup_price t
我正在实现代码以使用 JSON。 func setup() { if let flickrURL = NSURL(string: "https://api.flickr.com/
我尝试使用for循环声明变量,然后测试cols和rols是否相同。如果是,它将运行递归函数。但是,我在 javascript 中执行 do 时遇到问题。有人可以帮忙吗? 现在,在比较 col.1 和
我举了一个我正在处理的问题的简短示例。 HTML代码: 1 2 3 CSS 代码: .BB a:hover{ color: #000; } .BB > li:after {
我是一名优秀的程序员,十分优秀!