Angular 中的刷新令牌

保持用户会话流畅的关键在于无需持续登录。本文将演示如何在 angular 中实现令牌刷新流程,处理 401 错误并高效管理并发请求

什么是令牌刷新流程?

为了最大限度地降低安全风险,身份验证系统中的访问令牌通常具有较短的生命周期。当访问令牌过期时,刷新令牌允许应用程序在无需用户重新登录的情况下向服务器请求新的访问令牌。

Angular 实现

我们将使用 Angular 的 HTTP 拦截器实现令牌刷新机制。其目标是在重试原始请求之前拦截未授权的请求(401 错误)并刷新令牌。

完整流程

请求拦截:拦截器检测到 401 未授权响应。令牌刷新:如果令牌过期,则使用刷新令牌获取新令牌。请求重试:使用新令牌重试原始请求。队列管理:刷新令牌后,处理排队的请求。

Angular 中的刷新令牌

代码概述

令牌刷新逻辑:handleUnauthorized 方法负责在请求因令牌过期而失败时刷新令牌。

handleUnauthorized(req: HttpRequest, next: HttpHandler): Observable {  if (!this.isRefreshingToken) {    this.isRefreshingToken = true;    this.tokenSubject.next(null); // 通知所有等待请求令牌正在刷新    return this.refreshToken().pipe(      switchMap((newToken: string) => {        if (newToken) {          this.tokenSubject.next(newToken);          return next.handle(this.addToken(req, newToken)); // 使用新令牌重试原始请求        }        this.logout();        return throwError(() => '令牌过期');      }),      catchError((error) => {        this.logout();        return throwError(() => error);      }),      finalize(() => {        this.isRefreshingToken = false;      }),    );  } else {    // 令牌正在刷新时,将请求排队    return this.tokenSubject.pipe(      filter((token) => token !== null),      take(1),      switchMap((token) => next.handle(this.addToken(req, token))),    );  }}

handleUnauthorized 函数处理 HTTP 请求收到 401 未授权状态码的情况(指示访问令牌过期或无效),确保应用程序可以刷新令牌并无缝重试失败的请求。

防止多次刷新请求:该函数使用 isRefreshingToken 标志确保一次只发出一个令牌刷新请求。如果令牌正在刷新,则后续请求将排队,直到新令牌可用。

if (!this.isRefreshingToken) {  this.isRefreshingToken = true;  this.tokenSubject.next(null);}

刷新令牌:如果没有刷新请求正在进行,它将调用 refreshToken 方法来启动令牌刷新。收到新令牌后:它存储在 tokenSubject 中。使用更新后的令牌重试原始请求。

return this.refreshToken().pipe(  switchMap((newToken: string) => {    // ...  }));

处理并发请求:如果令牌刷新正在进行,该函数将后续请求排队。这些请求等待 tokenSubject 发出新令牌,然后继续。

return this.tokenSubject.pipe(  filter((token) => token !== null),  take(1),  switchMap((token) => next.handle(this.addToken(req, token))),);

错误处理:如果令牌刷新失败或抛出异常:用户将注销。错误将返回给调用者。

catchError((error) => {  this.logout();  return throwError(() => error);}),

清理:finalize 运算符确保重置 isRefreshingToken 标志,以便允许后续刷新请求。

finalize(() => {  this.isRefreshingToken = false;}),

将令牌添加到请求:addToken 方法将新令牌附加到传出请求的标头中。

addToken(request: HttpRequest, token: string): HttpRequest {  return request.clone({    setHeaders: {      'x-token': token,    },  });}

在 Angular HTTP 拦截器中使用:HTTP 拦截器是实现此流程的理想位置。它允许您拦截所有 HTTP 请求并全局处理令牌管理,而无需修改单个服务调用。

return next.handle(request).pipe(  catchError((error) => {    if (error.status === 401) {      return this.authService.handleUnauthorized(req, next);    }    return throwError(() => error);  }),);

总之,一个健壮的令牌刷新流程确保 Angular 应用程序拥有流畅的用户体验和安全的会话管理。通过有效处理 401 错误并管理并发请求,您可以提高应用程序的可靠性和用户满意度。

以上就是Angular 中的刷新令牌的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1500237.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
整洁代码的艺术:为什么它不仅仅是编写代码
上一篇 2025年12月19日 22:25:03
设计数据库架构日
下一篇 2025年12月19日 22:28:10

相关推荐

  • Golang空接口如何应用在项目中

    空接口可用于接收任意类型值,常见于日志函数、通用数据结构、JSON动态解析及配置驱动逻辑,提升代码灵活性,但需配合类型断言确保安全,避免滥用以降低维护成本。 空接口 interface{} 在 Go 语言中是一个非常灵活的类型,它可以存储任何类型的值。虽然它牺牲了一部分类型安全,但在实际项目中合理使…

    2026年5月10日
    100
  • Go语言接口与切片:如何识别和操作[]interface{}

    本文将深入探讨Go语言中如何识别和操作`[]interface{}`类型的切片。我们将介绍类型断言(Type Assertion)的关键作用,并通过`switch`语句演示如何安全地检测`[]interface{}`类型,并进而遍历其内部元素。文章旨在提供清晰的示例代码和专业指导,帮助开发者有效地处…

    2026年5月10日
    000
  • JavaScript计算器开发:解决数值显示与初始化问题

    本教程深入探讨了使用JavaScript构建计算器时常见的数值显示异常问题,特别是由于类属性未初始化导致的`Cannot read properties of undefined`错误。我们将详细分析问题根源,并通过在构造函数中调用初始化方法来解决该问题,同时优化显示逻辑,确保计算器功能稳定且界面显…

    2026年5月10日
    000
  • C++ 函数性能优化对系统稳定性的影响

    标题:C++ 函数性能优化对系统稳定性的影响 简介 函数性能优化是 C++ 程序员提高程序效率的关键技术。本文将探讨函数性能优化对系统稳定性的影响,并提供实战案例来证明这一点。 性能优化对稳定性的作用 立即学习“C++免费学习笔记(深入)”; 函数性能优化不仅可以提升程序速度,还可以提高系统的稳定性…

    2026年5月10日
    000
  • JS注解怎么和TypeScript结合_ JS注解在TypeScript环境下的应用

    TypeScript 支持通过配置 allowJs 和 checkJs 在 JavaScript 文件中识别 JSDoc 注解并进行类型检查,可在混合项目中提升类型安全;常见用法包括 @type、@param、@returns 和 @typedef,能为变量、函数参数等提供类型信息,支持与 .ts …

    2026年5月10日
    000
  • JavaScript动态下拉菜单:实现日期选项与价格计算关联

    在现代web应用中,动态生成表单元素并使其具备交互逻辑是常见的需求。特别是在需要根据用户选择调整价格或服务参数的场景下,下拉菜单()常被用来展示一系列选项。本教程将指导您如何利用javascript动态生成一个包含日期选项的下拉菜单,并为每个选项关联一个具体的数值(如剩余天数),进而实现一个基于用户…

    2026年5月10日
    000
  • Go语言中实现策略模式:灵活处理多源数据与格式转换

    本文探讨了如何在go语言中实现策略模式,以优雅地处理多源数据收集与多格式数据转换的场景。通过定义清晰的接口和具体的策略实现,结合go语言简洁的特性,展示了两种将策略集成到工作流中的方法,强调了go中接口驱动的灵活性。 在软件开发中,我们经常面临需要处理多种算法或行为,并根据具体情况选择其中之一的场景…

    2026年5月10日
    000
  • TypeScript函数体中如何高效判断参数类型?

    typescript 函数体中判断参数类型的技巧 typescript 中,我们可以定义接口来表示不同的数据类型。在本文中,我们将探讨如何在函数体中判断参数的类型,从而实现类型收窄,进行更精细的类型检查。 使用谓词函数 一种方法是编写谓词函数来手动检查类型。谓词函数返回的是 value is som…

    2026年5月10日
    000
  • JS如何实现策略模式

    策略模式通过封装算法使其可互换,JavaScript中利用函数作为一等公民实现,适用于表单验证等场景,结合工厂模式提升灵活性,但应避免过度设计。 策略模式的核心在于定义一系列算法,并将每一个算法封装起来,使它们可以相互替换。这使得算法可以在不影响客户端的情况下发生变化。在JS中,这可以通过函数作为一…

    2026年5月10日
    000
  • Svelte中正确导入数据与组件:避免常见误区

    在svelte开发中,理解如何正确导入数据和组件至关重要。svelte文件定义的是组件而非普通javascript模块,若需共享纯数据,应使用`.js`文件进行导出。本文将详细阐述svelte的导入机制,并通过示例代码展示如何区分导入数据与渲染组件,从而避免常见的导入错误,确保项目结构清晰且功能正确…

    2026年5月10日
    300
  • 如何通过GitHub API高效获取超过100个用户列表(分页教程)

    本教程旨在解决使用GitHub API获取用户列表时遇到的默认100个用户限制问题。我们将详细介绍两种主要的分页策略:利用Octokit库内置的paginate方法实现自动化分页,以及手动实现基于since参数的循环分页逻辑。文章将提供清晰的代码示例,并强调在不同场景下选择合适方法的注意事项,特别是…

    2026年5月10日
    100
  • 前端构建优化:利用常量折叠提升应用性能

    本文深入探讨了一种在构建阶段执行部分源代码以进行优化的技术——常量折叠(Constant Folding)。通过在编译时预计算表达式并替换为最终结果,该技术显著减少了运行时开销,提升了应用性能。文章将详细解释其工作原理、优势,并探讨其在现代前端构建工具中的应用与配置,旨在帮助开发者实现更高效的代码优…

    2026年5月10日
    000
  • 优化React-Redux应用中的用户与受保护数据按需加载

    本教程旨在解决React-Redux应用中用户数据和受保护API密钥在用户未登录时仍被请求,导致401错误的问题。通过引入条件性Redux状态初始化和动作分发逻辑,确保只有在用户被认为已认证时才发起相关的API请求,从而优化应用性能,减少不必要的网络流量和控制台错误。 在构建现代Web应用时,尤其是…

    2026年5月10日
    000
  • Go语言并发二叉树遍历:通道关闭与等价性判断的优雅方案

    本文探讨了在Go语言中并发遍历二叉树时,如何正确处理通道(channel)的关闭时机问题,尤其是在递归函数中。通过结合defer语句和闭包(closure)的巧妙运用,提供了一种优雅且健壮的解决方案,确保通道在所有值发送完毕后才被关闭,进而实现两个二叉树的等价性判断。 1. 并发遍历二叉树的需求与挑…

    2026年5月10日
    000
  • 动态语言中静态类型的讽刺

    c++kquote>您也可以在 medium 上阅读这篇文章。 当我们看到编程语言如何随着时间的推移而演变时,总是很有趣。 曾几何时,当我开始进入软件开发世界时,python、php 和 javascript 等动态语言因其灵活性和适合快速开发的简洁语法而受到赞赏。 然而,随着这些弱类型语言的…

    2026年5月10日
    000
  • 为什么 TypeScript 比 JavaScript 更好

    javascript 长期以来一直是 web 开发的基石,支持从小型脚本到大型应用程序的各种项目。然而,随着项目规模的扩大,javascript 的动态类型和缺乏结构性可能会成为开发的瓶颈。typescript 应运而生,它凭借静态类型检查和强大的工具集,迅速成为许多开发者构建可靠、可扩展应用程序的…

    2026年5月10日
    100
  • 深入解析Angular中循环计算与数组操作的常见陷阱及优化实践

    本文深入探讨了angular应用中处理循环计算和动态数组时常见的逻辑错误。通过一个租金计算器示例,我们分析了`for`循环中未能正确累加迭代值以及数组填充不当的问题,并提供了详细的解决方案,包括优化计算逻辑、正确使用数组`push`方法,以及遵循typescript和javascript的最佳实践,…

    2026年5月10日
    000
  • Golang环境变量配置自动化脚本方法

    答案:通过编写Shell脚本自动化配置Go环境变量,可实现GOROOT、GOPATH、GOBIN及PATH等变量的自动设置,提升开发效率。具体做法是创建setup_go_env.sh脚本,定义GOROOT为Go安装路径(如/usr/local/go),GOPATH为工作区(如~/go_project…

    2026年5月10日
    100
  • js怎样实现文件拖拽上传 js文件拖拽上传的4步完整实现

    文件拖拽上传的核心步骤是监听拖拽事件、获取文件信息和发送文件到服务器。具体为:1. 监听dragover和drop事件;2. 通过event.datatransfer.files获取文件列表;3. 使用formdata结合xmlhttprequest或fetch api上传文件。优化用户体验需注意:…

    2026年5月10日
    000
  • 深入理解Go语言中的短声明:=与长声明var

    Go语言提供了两种主要的变量声明和初始化方式:短声明:=和长声明var。:=主要用于函数内部,实现变量的声明与初始化,并常用于控制流语句中以限制变量作用域,例如在if语句中处理错误。而var则更为通用,可用于包级别或函数内部,支持显式类型声明、不带初始化的声明以及批量声明,提供了更大的灵活性。 1.…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信