- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
内容:
我们的Web应用程序可以同时显示不同的帮助面板。
当面板要求给定ID(例如help-id-1
)时,我们将到达一个API,并传递该ID,然后获得所需的帮助。
现在,由于某些原因,我们可能会在同一帮助面板上显示两次或更多次。但是,我们当然不希望为同一项目多次击中该API,如果没有错误或当前正在获取该API。
我们的“生产者”为我们提供了检索它的冷流:
const getHelpContentById = (id: string) => fromPromise(
httpCallToGetHelpResultFromThirdLib(id)
).pipe(
catchError(error => of({ status: 'ERROR', item: null, id })),
// extracting the body of the response
map(getHelpItemFromResponse),
// wrapping the response into an object { status: 'SUCCES', item, id }
map(item => setStatusOnHelpItem(item, id)),
startWith({ status: 'LOADING', item: null, id }),
)
SUCCESS
或
ERROR
。
LOADING
),则该订阅应获得与第一次调用相同的流,而无需再次获取API
help-id-1
失败,则不应关闭该流,而应关闭其中的
next
类型为
{ status: 'ERROR', item: null, id }
的值,这样,如果另一个组件尝试再次显示
help-id-1
,由于该ID的最后状态为
ERROR
,因此它应尝试再次访问API,并且两个订阅者都应收到实时更新
{ status: 'LOADING', item: null, id }
,然后返回错误或成功
private helpItems: Map<
string,
{ triggerNewFetchForItem: () => void; obs: Observable<HelpItemWithStatus> }
> = new Map();
private getFromCacheOrFetchHelpItem(id: string): Observable<HelpItemWithStatus> {
let triggerNewFetchForItem$: BehaviorSubject<HelpItemWithStatus>;
const idNotInCache = !this.helpItems.has(id);
if (idNotInCache) {
triggerNewFetchForItem$ = new BehaviorSubject<HelpItemWithStatus>(null);
this.helpItems.set(id, {
triggerNewFetchForItem: () => triggerNewFetchForItem$.next(null),
obs: triggerNewFetchForItem$.pipe(
switchMap(() => getHelpContentById(id)),
shareReplay(1),
),
});
return this.helpItems.get(id).obs;
} else {
return this.helpItems.get(id).obs.pipe(
tap(item => {
if (item.status === ContentItemStatus.ERROR) {
this.helpItems.get(id).triggerNewFetchForItem();
}
})
);
}
}
public getHelpItemById(id: string): Observable<HelpItemWithStatus> {
return this.getFromCacheOrFetchHelpItem(id);
}
private getFromCacheOrFetchHelpItem4(id: string): Observable<HelpItemWithStatus> {
let item = this.items.get(id);
if (item && item.status !== ContentItemStatus.ERROR) {
return of(item);
}
return getNewWrappedHelpItem(this.contentfulClient, id).pipe(
tap(item => this.items.set(id, bs),
shareReplay(1)),
)
}
Map
)。最好的办法显然是为此配备一个新的运算符:)
最佳答案
我基本上在工作相同的场景。我已经考虑了一段时间,但最终决定花点时间。我的解决方案有些粗糙,但是我会在优化时尝试更新答案。我希望能有更好的方法来提供反馈。
问题/步骤
这是一个相当复杂的问题,所以我将逐步进行构建。为了简单起见,我们将从不带参数的api方法开始。
共享流,以便多个订阅者不会触发多个api请求
只需在最后添加share()
运算符即可使其成为多播。
return api().pipe(share());
share()
更改为
shareReplay(1)
。该参数指示要共享的先前响应的数量。我们只希望发出最后一个,所以我们放入
1
。
tap
运算符保留对最后一个发出的值的引用,并执行
of(data)
而不是如果最后一个成功,则不返回流。仅当您不再希望再次调用api时才适用(如所讨论的OP),但我将其保持通用性以使其灵活地适用于其他解决方案。
return api().pipe(shareReplay(1));
retryWhen
运算符来实现自动化,则我们需要某种手动方式,例如加载新组件。加载新组件时,它们会请求流,因此可以正常工作。
ReplaySubject
或
BehaviorSubject
来避免超时,但这样做时(
ExpressionChangedAfterItHasBeenCheckedError
)遇到了角度变化检测的问题。我需要更深入地研究它。
share
在外部流上。我们想分享它而不是内在的。另请注意,由于我们每次都需要一个新的内部流,因此我使用的是
switchMap
而不是
switchMapTo
。
const trigger = new Subject<void>();
setTimeout(() => trigger.next());
return trigger.pipe(
switchMap(() => api()),
shareReplay(1)
);
catchError
运算符可让您返回可观察值。由于我们希望将其作为消息,因此我们只执行
catchError(e => of(e))
。问题在于这将结束流。解决方法是将捕获的内容放入
switchMap
的内部,以便内部流可以死掉而外部流可以继续前进。
return trigger.pipe(
switchMap(() => api().pipe(
catchError(err => of(err))
),
shareReplay(1)
);
startWith
运算符发送获取通知(因此为什么会结束)。
return trigger.pipe(
switchMap(() => api().pipe(
map((data) => ({ state: 'SUCCESS', data })),
catchError(err => of({ state: 'FAILURE', err })),
startWith({ state: 'FETCHING' })
),
shareReplay(1)
);
distinct
运算符与触发器配合使用以在api调用解析后将其重置。第二种方法很棘手,因为在构造流时需要引用该流。因此,我们将只使用一个变量,并且可以将
trigger.next()
包装在if中,也可以在流上放置一个过滤器。我将做一个过滤器。
private state: string;
...
return trigger.pipe(
filter(() => this.state !== 'FETCHING'),
switchMap(() => api().pipe(
map((data) => ({ state: 'SUCCESS', data })),
catchError(err => of({ state: 'FAILURE', err })),
startWith({ state: 'FETCHING' }),
tap(x => { this.state = x.state; })
),
shareReplay(1)
);
...
filter(() => this.state == null || this.state === 'FAILURE'),
...
import { Observable } from 'rxjs/Observable';
import { Subject } from 'rxjs/Subject';
import { tap, switchMap, map, startWith, catchError, shareReplay, filter } from 'rxjs/operators';
import { of } from 'rxjs/observable/of';
// posible states of the api request
export enum ApiStateType {
Fetching,
Success,
Failure
}
// wrapper for the api status messages
export interface ApiStatus<T> {
state: ApiStateType;
params: any[],
data: T
}
// information related to a stream for a unique set of parameters
interface StreamConfig<T> {
state: ApiStateType;
trigger: Subject<void>;
stream: Observable<ApiStatus<T>>;
}
export function generateCachedApi<T>(
api: (...params) => Observable<T>,
generateKey: (...params) => string
): (...params) => Observable<ApiStatus<T>> {
const cache = new Map<string, StreamConfig<T>>();
return (...params): Observable<ApiStatus<T>> => {
const key = generateKey(...params);
let config = cache.get(key);
if (!config) {
console.log(`created new stream (${key})`);
config = <StreamConfig<T>> { trigger: new Subject<void>() };
config.stream = config.trigger.pipe(
filter(() => config.state == null || config.state === ApiStateType.Failure),
switchMap(() => {
return api(...params).pipe(
map((data) => (<ApiStatus<T>>{ state: ApiStateType.Success, params, data })),
catchError((data, source) => of(<ApiStatus<T>>{ state: ApiStateType.Failure, params, data })),
startWith(<ApiStatus<T>>{ state: ApiStateType.Fetching, params }),
tap(x => { config.state = x.state; })
);
}),
tap(x => { console.log('PUBLISH', x)}),
shareReplay(1),
);
cache.set(key, config);
} else {
console.log(`returned existing stream (${key})`);
}
setTimeout(() => { config.trigger.next() });
return config.stream;
}
}
cacheMap
运算符来尝试做到这一点。我有一个发出api参数的源,
cacheMap
运算符将为唯一的一组参数查找或创建流,并返回其
mergeMap
样式。问题在于,现在每个订阅者都将订阅该内部可观察对象。因此,您必须添加一个过滤器(请参见下面的替代解决方案)。
FETCHING
到
SUCCESS
的所有内容,这可能会导致额外的处理,尽管用户可能不会注意到。理想情况下,我们将有一个
replayByKey
运算符,但我还没有写出来。因此,现在我仅使用地图。使用地图的问题在于,我们仍然向已经收到地图的订户发送相同的值。因此,我们将
distinctUntilChanged
运算符添加到实例流。或者,您可以创建实例流,然后在其上放一个
takeUntil
,触发器是为成功进行过滤的实例流,并放一个
delay(0)
以允许最后一个值在关闭之前通过管道。这样就可以完成数据流,因为一旦成功就永远不会获得新的值,就可以了。我采用了与众不同的方法,因为如果您想更改其要求,它可以使您获得新的价值。
mergeMap
而不是
switchMap
,因为我们可以同时进行针对不同参数的运行中请求,并且我们不想取消针对不同参数的请求。
import { Observable } from 'rxjs/Observable';
import { Subject } from 'rxjs/Subject';
import { tap, mergeMap, map, startWith, catchError, share, filter, distinctUntilChanged } from 'rxjs/operators';
import { of } from 'rxjs/observable/of';
// posible states of the api request
export enum ApiStateType {
Fetching,
Success,
Failure
}
// wrapper for the api status messages
export interface ApiStatus<T> {
state: ApiStateType;
key: string;
params: any[];
data: T;
}
export function generateCachedApi<T>(
api: (...params) => Observable<T>,
generateKey: (...params) => string
): (...params) => Observable<ApiStatus<T>> {
const trigger = new Subject<any[]>();
const stateCache = new Map<string, ApiStatus<T>>();
const stream = trigger.pipe(
map<any[], [any[], string]>((params) => [ params, generateKey(...params) ]),
tap(([_, key]) => {
if (!stateCache.has(key)) {
stateCache.set(key, <ApiStatus<T>> {})
}
}),
mergeMap(([params, key]) => {
const apiStatus = stateCache.get(key);
if (apiStatus.state === ApiStateType.Fetching || apiStatus.state === ApiStateType.Success) {
return of(apiStatus);
}
return api(...params).pipe(
map((data) => (<ApiStatus<T>>{ state: ApiStateType.Success, key, params, data })),
catchError((data, source) => of(<ApiStatus<T>>{ state: ApiStateType.Failure, key, params, data })),
startWith(<ApiStatus<T>>{ state: ApiStateType.Fetching, key, params }),
tap(state => { stateCache.set(key, state); })
)
}),
tap(x => { console.log('PUBLISH', x)}),
share()
);
return (...params): Observable<ApiStatus<T>> => {
const key = generateKey(...params);
const instanceStream = stream.pipe(
filter((response) => response.key === key),
distinctUntilChanged()
);
setTimeout(() => { trigger.next(params) });
return instanceStream;
}
}
关于angular - RxJs难题:共享流并在有新订阅时有条件地重试,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49069193/
RxJS 中是否有一个运算符可以让我缓冲项目并在信号可观察对象触发时将它们一个一个地放出?有点像bufferWhen,但不是在每个信号上转储整个缓冲区,而是每个信号转储一定数量。它甚至可以转储信号 o
我正在像这样创建我的可观察源(每 5 秒调用一次 api): const obs$ = Observable.interval(5000).switchMap(() => makeApiCall())
我有一个 Action ,然后将触发一个ajax请求。 如果由于某种原因操作失败,我什么也不想做。我没有执行可以执行的无操作功能,而不是创建只返回先前状态的空白操作? export default f
在以下代码中:- RxJS.Observable.of(1,2).first().subscribe((x) => console.log(x);); 给定运营商 first() 是否有必要取消订阅?
我有一种情况,可以在很短的时间内将很多事件发送到流中。我想要一个运算符,它是ojit_code和debounceTime的混合体。 以下演示可用于说明我想拥有的https://stackblitz
我的用例如下:我得到事件,有时会突然发生。如果发生突发,我只需要处理一次即可。去抖动会执行此操作。 但是,去抖动仅给我提供连拍的最后一个元素,但我需要了解连拍中的所有元素才能汇总(使用平面图)。 这可
简化以下代码示例的方法是什么? 我找不到合适的运算符..有人可以举一个简短的例子吗? this.returnsObservable1(...) .subscribe( success =>
在RxJS 6中,如何导入静态合并功能以合并Observable列表? 我希望能够做到: const merged$ = merge( obs1$, obs2$, obs3$
我正在阅读 RxJS 的官方文档,然后我意识到它们都在做完全相同的事情。 对我来说,它们看起来完全相似。 如果有区别请指出。 最佳答案 我将根据它们的 Time 版本来描述它们之间的区别,因为这是我最
我对基本的 RxJS 概念有点熟悉,比如 Observables、Observers 和 Subjects,但是 RxJS Notifications概念对我来说是全新的。 它有什么用?我应该什么时候
从 rxjs 6.5 切换到 rxjs 7 后,我遇到了这个奇怪的错误。我不确定这是 rxjs 7 的类型问题还是 stackblitz ( https://stackblitz.com/edit/r
以前我只能使用此代码导入使用过的运算符: import 'rxjs/Observable'; import 'rxjs/add/operator/map'; import 'rxjs/add/oper
combineLatest 函数可以从 rxjs 和 rxjs/operators 导入。 当我从 rxjs/operators 导入它时(就像我导入 combineAll 我收到以下错误: TS23
我有一系列事件通过 fromEventPattern 进行像这样: fromEventPattern(addEventHandler).subscribe(ps$); 由于业务怪癖,我预计有时会抛出异
我是 rxjs 的新手,无法解决这个问题: 我有两个流: 一个有传入的对象 ---a----b----c----d-----> 一个是从列表中选择的对象 ----------------c---->
如果一个 observable 完成,我是否仍然需要取消订阅/处置(在 RxJS 中)该 observable 以删除 Observer(防止内存泄漏),或者一旦 onComplete 或 onErr
我有这样的订阅: this.test.subscribe(params => { ...some code }); 如果我传递回调函数而不是箭头函数,则缺少上下文。 我想将上下文绑定(bind)到
我有一个可观察的: messages: string[] = ['a', 'b', 'c']; const source = from(messages) 你如何延迟它,所以当有人订阅它时,它
我可以让 observable 触发一次该值。但我希望它在变量的值发生变化时发生。实际上我需要一个观察者。这就是我认为 observable 的意义所在。观察事物的值(value)或状态并更新订阅它的
我有以下代码: const fetchBook = (bookId: number) => { const title = 'Book' + bookId; console.log('
我是一名优秀的程序员,十分优秀!