- 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/
我正在用 Java 创建一组小部件,用于解码和显示在串行接口(interface)接收到的消息。 消息类型由唯一标识符定义。每个小部件只对特定标识符感兴趣。 如何对应用程序进行编程,以便将消息正确分发
我有以下代码,其中包含多个订阅。我需要实现的是这样的: 订阅activateRoute 以获取用户和产品数据。 返回商品数据后,使用商品数据订阅getSeller服务。 使用返回的卖家数据订阅 get
我已经使用 Fitbit 的 PHP 库 (www.disciplinexgames.com/fitbit) 在我的网站中成功集成了 FitBit api。它工作正常,但我现在想使用订阅 API,以便
在我的 Angular 7 应用程序中,我有下一个功能: getUserData(uid) { return this.fireStore.collection('users').doc(
我正在尝试在 Node 中实现发布/订阅模式,但不使用 Redis。功能应该是相同的;您可以发布到 channel ,订阅 channel 并收听数据(如果您已订阅);以下是 Redis 功能: pu
这是我当前的应用程序结构: /client/client.js /server/server.js collection.js 有 HTML 和 CSS 文件,但这些与我的问题无关。在将我的应用程序拆
我们正在使用OpenTok建立视频聊天室体验,并且在基本工作正常的同时,我发现当 session 室中有很多参与者发布音频时,本底噪声非常高。像Zoom这样的浏览器外解决方案似乎没有这种高水平的“白噪
RabbitMQ 是点对点还是发布-订阅?或者两者都取决于配置选项? 我一直在查看配置,它们似乎都支持点对点模型而不是发布-订阅。即消息一旦被消费就会从队列中删除,并且不可用于第二个消费者。 最佳答案
我是 Angular 6 和 ngrx 商店的新人。我尝试在从商店订阅数据后调度操作,但它会导致无限循环并使浏览器崩溃?我错了什么。我发现它使用 rxjs 的 do/tap 运算符但仍然不起作用的一些
这个问题已经有答案了: Property '...' has no initializer and is not definitely assigned in the constructor (37
这个问题已经有答案了: Property '...' has no initializer and is not definitely assigned in the constructor (37
我正在使用 Visual Studio 2017 v15.6.2 和 Azure Services Authentication Extension 为支持 MSI 的应用程序进行本地 azure 功
我想知道如何确定给定的 WC_Product 对象 $product 是否是订阅产品。 最佳答案 您可以使用他们的辅助函数,这可能是最完整的: if( class_exists( 'WC_Subscr
我正在研究使用服务器发送的事件作为支持 api 来实现“订阅”类型。 我正在苦苦挣扎的是接口(interface),更准确地说,是这种操作的 http 层。 问题: 使用原生 EventSource不
我会根据每个用户的订阅类型向我的用户发送通知。 例如: 用户 A 订阅了所有新闻文章 用户 B 订阅了所有评论 用户 C 订阅了网站上的所有新内容 我有一个每 5 分钟运行一次的脚本(除非该脚本仍在运
我正在使用 Ionic2/Angular2,并且需要使用参数 authData 调用函数,如下所示。 public auth: FirebaseAuth this.auth.subscrib
已结束。此问题正在寻求书籍、工具、软件库等的推荐。它不满足Stack Overflow guidelines 。目前不接受答案。 我们不允许提出寻求书籍、工具、软件库等推荐的问题。您可以编辑问题,以便
我们现有的系统可以持续处理大量文件。粗略地说,每天大约有 300 万个文件,大小从几千字节到超过 50 MB。这些文件从接收到完成使用会经历几个不同的处理阶段,具体取决于它们所采用的路径。由于这些文件
我有一项服务,我使用 Paypal 订阅。 Paypal 有 webhooks。问题是我不知道我需要使用哪个,不知道用户是否为下个月付款。 我使用了 Billing subscription rene
我目前正在为一个网站整理一个处理脚本,遇到了一个我似乎无法找到明确答案的问题。 Paypal 的文档充其量是不确定的,我对 Paypal 的使用还不够多,无法从他们提供的信息中轻松辨别答案。 当通过
我是一名优秀的程序员,十分优秀!