新闻中心
Angular Guard 中 combineLatest 的正确使用姿势

本文旨在解决 Angular 应用中使用 combineLatest 结合多个 Observable 实现路由守卫时,可能出现的逻辑错误问题。通过分析一个实际案例,我们将深入探讨如何正确地使用 combineLatest,避免不必要的页面跳转,并提供清晰的代码示例和注意事项,帮助开发者构建更健壮的路由守卫。
在使用 Angular 开发应用时,路由守卫(Guards)是控制用户访问特定路由的重要机制。当需要根据多个条件(例如:用户权限、数据状态等)来决定是否允许用户访问某个路由时,常常会用到 RxJS 的 combineLatest 操作符。然而,不当的使用可能导致意料之外的行为,例如不正确的页面重定向。
问题分析
假设我们有一个场景:需要创建一个路由守卫 CanCreateNewCv,用于控制用户是否可以访问 /new 页面。该守卫需要同时检查两个条件:
- 用户是否已经创建了 CV(简历)。
- 用户是否是管理员或经理。
如果用户已经创建了 CV,并且不是管理员或经理,则不允许访问 /new 页面,并重定向到 /list 页面。如果用户是管理员或经理,则允许访问 /new 页面,无论是否已经创建了 CV。
以下是可能存在问题的代码:
import { Injectable } from '@angular/core';
import { CanActivate, Router } from '@angular/router';
import { Observable, combineLatest } from 'rxjs';
import { map, finalize } from 'rxjs/operators';
import { PersonsService } from '../services/persons.service';
import { AdministrationService } from '../services/administration.service';
import { CustomSnackbarService } from '../services/custom-snackbar.service';
@Injectable({
providedIn: 'root',
})
export class CanCreateNewCv implements CanActivate {
constructor(
private usersService: PersonsService,
private router: Router,
private administrationService: AdministrationService,
private snackbarService: CustomSnackbarService
) {}
canActivate(): Observable<boolean> | boolean | Promise<boolean> {
let isAllowed = false;
const cv$ = this.usersService.getPersonsByPageAndFilter(10, 0).pipe(
map((data) => {
if (data.allDataCount > 0) {
this.router.n*igateByUrl('/list');
return true;
}
return false;
})
);
const admin$ = this.administrationService.getCurrentUser().pipe(
map((currentUser) => {
if (currentUser.isAdmin || currentUser.isManager) {
return true;
}
return false;
})
);
return combineLatest([cv$, admin$], (isCvUploaded, isAdminOrManager) => {
isAllowed = isAdminOrManager ? true : isCvUploaded ? false : true;
return isAllowed;
}).pipe(
finalize(() => {
if (!isAllowed)
this.snackbarService.open(
'This profile has CV already created!',
'Info'
);
})
);
}
}上述代码的问题在于,无论用户是否是管理员或经理,cv$ Observable 都会尝试重定向到 /list 页面。这导致即使管理员或经理访问 /new 页面,也会被重定向。
UXbot
AI产品设计工具
185
查看详情
解决方案
要解决这个问题,需要将页面重定向的逻辑移到 combineLatest 的回调函数中,只有当 isAllowed 为 false 时才进行重定向。同时,简化 cv$ 和 admin$ Observable 的 map 操作,直接返回布尔值。
以下是修改后的代码:
import { Injectable } from '@angular/core';
import { CanActivate, Router } from '@angular/router';
import { Observable, combineLatest } from 'rxjs';
import { map, finalize } from 'rxjs/operators';
import { PersonsService } from '../services/persons.service';
import { AdministrationService } from '../services/administration.service';
import { CustomSnackbarService } from '../services/custom-snackbar.service';
@Injectable({
providedIn: 'root',
})
export class CanCreateNewCv implements CanActivate {
constructor(
private usersService: PersonsService,
private router: Router,
private administrationService: AdministrationService,
private snackbarService: CustomSnackbarService
) {}
canActivate(): Observable<boolean> | boolean | Promise<boolean> {
let isAllowed = false;
const cv$ = this.usersService
.getPersonsByPageAndFilter(10, 0)
.pipe(map((data) => data.allDataCount > 0));
const admin$ = this.administrationService
.getCurrentUser()
.pipe(map((currentUser) => currentUser.isAdmin || currentUser.isManager));
return combineLatest([cv$, admin$], (isCvUploaded, isAdminOrManager) => {
isAllowed = isAdminOrManager ? true : isCvUploaded ? false : true;
if (!isAllowed) {
this.router.n*igateByUrl('/list');
}
return isAllowed;
}).pipe(
finalize(() => {
if (!isAllowed)
this.snackbarService.open(
'This profile has CV already created!',
'Info'
);
})
);
}
}在这个修改后的版本中,cv$ Observable 现在直接返回一个布尔值,指示用户是否已经上传了 CV。admin$ Observable 也做了类似的简化,直接返回一个布尔值,指示用户是否是管理员或经理。
关键的改变在于 combineLatest 的回调函数中,只有当 isAllowed 为 false 时,才会调用 this.router.n*igateByUrl('/list') 进行页面重定向。这样确保了只有当用户不是管理员或经理,并且已经上传了 CV 时,才会被重定向到 /list 页面。
注意事项
- 避免在 Observable 中进行副作用操作: 尽量避免在 Observable 的 map 操作符中直接进行页面重定向等副作用操作。将这些操作移到 combineLatest 的回调函数中,可以更好地控制执行时机。
- 理解 combineLatest 的行为: combineLatest 会在所有输入 Observable 都发出至少一个值后,才会发出值。因此,确保所有输入 Observable 都会发出值,否则守卫可能无法正常工作。
- 错误处理: 在实际应用中,需要考虑错误处理的情况。如果 getPersonsByPageAndFilter 或 getCurrentUser 方法抛出错误,可能会导致守卫无法正常工作。可以使用 RxJS 的 catchError 操作符来处理这些错误。
总结
通过本文的分析,我们了解了如何正确地使用 combineLatest 结合多个 Observable 实现 Angular 路由守卫。关键在于理解 combineLatest 的行为,并避免在 Observable 中进行不必要的副作用操作。通过合理地组织代码,可以构建出更健壮、更可靠的路由守卫。
以上就是Angular Guard 中 combineLatest 的正确使用姿势的详细内容,更多请关注其它相关文章!
# 无法正常
# 罗湖区网站优化推广费用
# 网站优化前的英雄
# 楚雄网站建设报价
# 网站优化外包怎么收费的
# seo多久能见效果
# 荔湾网站推广优化方案
# 凌海响应式网站建设
# 国外推广中国游戏的网站
# 长治网站开发推广公司
# 衢州营销推广价格贵吗
# 服务端
# 布尔值
# js
# 传了
# 如何使用
# 移到
# 才会
# 多个
# 重定向
# 回调
# gate
# 简历
# 路由
# 回调函数
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
Lar*el的路由模型绑定怎么用_Lar*el Route Model Binding简化控制器逻辑
魅族17怎样用浏览器译外语网页_iPhone魅族17浏览器译外语网页【即时翻译】
海棠账号登录入口_登录海棠账户同步阅读记录
Golang如何实现微服务鉴权与权限控制_Golang微服务鉴权与权限管理实践
CSS条件样式无法按设备触发怎么排查_media条件语句正确设置解决触发问题
高德地图总提示网络异常怎么办 高德地图离线导航设置与网络排查方法
离线运行Go语言之旅:本地部署与GOPATH配置指南
谷歌浏览器怎么给标签页静音_Chrome标签静音快捷操作
PHP表单数据传递:如何通过隐藏输入字段获取动态ID
苹果手机如何防止被恶意App追踪
外媒分析《GTA6》定价:卖100美元可以但真没必要!
12306选座怎么选到临时改签座_12306改签选座策略与步骤
千牛数据看板网页版_千牛数据看板网页版访问方法
PDF文件体积过大处理_PDF压缩技巧详解
抖音创作助手登录入口_抖音创作辅助工具官网直达
Go语言中高效处理x-www-form-urlencoded表单数据
多闪网页版在线观看免费入口_多闪官网访问入口
AO3访问入口汇总 AO3网页版同人作品一键直达
百度浏览器字体显示异常偏小_百度浏览器字体渲染修复方案
c++如何实现一个简单的软件渲染器_c++从零开始的3D图形学
浏览器打开即用 美图秀秀网页版入口
解决深度学习模型训练初期异常高损失与完美验证准确率问题
AO3官网镜像链接 Archive of Our Own同人文在线浏览
文本文档写html代码怎么运行_文本文档html代码运行步骤【教程】
C++ typeid如何获取类型信息_C++ RTTI运行时类型识别用法
谷歌学术网站直达地址 谷歌学术搜索网页版一键进入
可靠CSGO开箱平台解析 CSGO开箱网合集
Python字典中优雅地迭代剩余元素的方法
Composer的 "conflict" 字段有什么用_如何声明不兼容的包以避免依赖冲突
React/Next.js中实现列表项的动态移动与状态管理:兼论唯一键的重要性
Lar*el 递归关系中排除指定分支的教程
Excel中VLOOKUP的第四个参数是干什么用的_Excel VLOOKUP第四参数作用解析
我的世界mc.js免费游戏直接能玩 我的世界mc.js小游戏免费秒玩入口
蛙漫正版漫画平台入口_蛙漫免费阅读全站漫画资源
高德地图家和公司地址在哪设置 高德地图通勤路线设置方法【超详细】
c++中的std::basic_string的SSO优化_c++短字符串优化深度解析
解决移动端滚动问题的overflow属性应用指南
c++如何使用std::memory_order控制原子操作顺序_c++ C++11内存模型详解
印象笔记怎样用批量导出备知识库_印象笔记用批量导出备知识库【备份方法】
C++如何操作注册表_Windows平台下C++读写注册表的API函数详解
韩小圈电脑版在线入口_网页版免费登录地址
Eclipse怎么运行工程_Eclipse工程运行配置说明
格力空气能E5故障代码是什么情况_格力空气能E5代码解析与应对措施
Go语言中Map值调用指针接收器方法的限制与应对
C++指针和引用有什么区别_C++内存管理核心概念深度解析
CSS布局:解决全屏元素100%尺寸与外边距导致的页面溢出问题
Angular响应式表单:实现提交后表单及按钮的禁用与只读化
J*aScript中localStorage数据的获取、清洗与格式化教程
J*aScript中向JSON对象添加新属性的正确姿势
夸克浏览器桌面版同步不了书签怎么处理 夸克浏览器跨设备同步异常解决方案


2025-10-06
浏览次数:次
返回列表
}).pipe(
finalize(() => {
if (!isAllowed)
this.snackbarService.open(
'This profile has CV already created!',
'Info'
);
})
);
}
}