当前位置: 首页 > wzjs >正文

什么是网站建设?小型建筑公司有哪些

什么是网站建设?,小型建筑公司有哪些,秦皇岛网站关键词,厦门商城网站开发在前后端分离的架构中,双 Token 无感刷新是一种常见的身份验证机制,用于在 Access Token 过期时,通过 Refresh Token 自动获取新的 Access Token,从而避免用户频繁登录。 1. 双 Token 无感刷新的核心流程 1.1 核心流程 用户登录&…

在前后端分离的架构中,双 Token 无感刷新是一种常见的身份验证机制,用于在 Access Token 过期时,通过 Refresh Token 自动获取新的 Access Token,从而避免用户频繁登录。


1. 双 Token 无感刷新的核心流程

1.1 核心流程

  1. 用户登录

    • 前端发送用户名和密码到后端。
    • 后端验证成功后,返回 Access TokenRefresh Token
    • 前端将这两个 Token 存储在本地(如 localStorageCookie)。
  2. Access Token 过期

    • 前端发起请求时,携带 Access Token
    • 如果 Access Token 过期,后端返回 401 Unauthorized 错误。
  3. 自动刷新 Token

    • 前端检测到 401 错误后,使用 Refresh Token 请求新的 Access Token
    • 后端验证 Refresh Token 的有效性,如果有效,返回新的 Access Token
    • 前端更新本地存储的 Access Token,并重试之前的请求。
  4. Refresh Token 过期

    • 如果 Refresh Token 也过期,后端返回 401 错误。
    • 前端清除本地存储的 Token,并跳转到登录页面。

2. 前端实现(React + TypeScript)

2.1 配置 Axios 拦截器

Axios 拦截器用于在请求发送前和响应返回后执行逻辑,是实现无感刷新的核心。

import axios, { AxiosInstance, AxiosRequestConfig, AxiosResponse, AxiosError } from 'axios';// 创建 Axios 实例
const api: AxiosInstance = axios.create({baseURL: 'https://your-api-url.com',headers: {'Content-Type': 'application/json',},
});// 请求拦截器:在请求头中添加 Access Token
api.interceptors.request.use((config: AxiosRequestConfig) => {const accessToken = localStorage.getItem('accessToken');if (accessToken) {config.headers.Authorization = `Bearer ${accessToken}`;}return config;},(error: AxiosError) => {return Promise.reject(error);}
);// 响应拦截器:处理 Access Token 过期
api.interceptors.response.use((response: AxiosResponse) => {return response;},async (error: AxiosError) => {const originalRequest = error.config;// 检测到 401 错误且未重试过if (error.response?.status === 401 && !originalRequest._retry) {originalRequest._retry = true; // 标记请求已重试try {// 使用 Refresh Token 获取新的 Access Tokenconst refreshToken = localStorage.getItem('refreshToken');const response = await axios.post('https://your-api-url.com/refresh-token', { refreshToken });const { accessToken } = response.data;// 更新本地存储的 Access TokenlocalStorage.setItem('accessToken', accessToken);api.defaults.headers.common['Authorization'] = `Bearer ${accessToken}`;originalRequest.headers['Authorization'] = `Bearer ${accessToken}`;// 重试原始请求return api(originalRequest);} catch (refreshError) {// 刷新 Token 失败,清除本地存储并跳转到登录页localStorage.removeItem('accessToken');localStorage.removeItem('refreshToken');window.location.href = '/login';return Promise.reject(refreshError);}}return Promise.reject(error);}
);export default api;

2.2 使用 React Context 管理 Token 状态

通过 React Context 管理 Token 的状态,方便在组件中共享和更新。

import React, { createContext, useContext, useState, useEffect } from 'react';interface AuthContextType {accessToken: string | null;refreshToken: string | null;setTokens: (accessToken: string, refreshToken: string) => void;clearTokens: () => void;
}const AuthContext = createContext<AuthContextType | undefined>(undefined);export const AuthProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {const [accessToken, setAccessToken] = useState<string | null>(localStorage.getItem('accessToken'));const [refreshToken, setRefreshToken] = useState<string | null>(localStorage.getItem('refreshToken'));const setTokens = (accessToken: string, refreshToken: string) => {localStorage.setItem('accessToken', accessToken);localStorage.setItem('refreshToken', refreshToken);setAccessToken(accessToken);setRefreshToken(refreshToken);};const clearTokens = () => {localStorage.removeItem('accessToken');localStorage.removeItem('refreshToken');setAccessToken(null);setRefreshToken(null);};return (<AuthContext.Provider value={{ accessToken, refreshToken, setTokens, clearTokens }}>{children}</AuthContext.Provider>);
};export const useAuth = () => {const context = useContext(AuthContext);if (!context) {throw new Error('useAuth must be used within an AuthProvider');}return context;
};

2.3 在组件中使用

在需要身份验证的组件中,使用 useAuth Hook 获取和更新 Token。

import React from 'react';
import { useAuth } from './AuthContext';
import api from './api';const ProtectedComponent: React.FC = () => {const { accessToken, setTokens, clearTokens } = useAuth();const fetchData = async () => {try {const response = await api.get('/protected-resource');console.log(response.data);} catch (error) {console.error('Failed to fetch data', error);}};return (<div><button onClick={fetchData}>Fetch Data</button><button onClick={clearTokens}>Logout</button></div>);
};export default ProtectedComponent;

3. 后端实现(Node.js + Express)

3.1 登录接口

登录成功后返回 Access TokenRefresh Token

const express = require('express');
const jwt = require('jsonwebtoken');
const app = express();app.use(express.json());const SECRET_KEY = 'your-secret-key';
const REFRESH_SECRET_KEY = 'your-refresh-secret-key';app.post('/login', (req, res) => {const { username, password } = req.body;// 模拟用户验证if (username === 'admin' && password === 'password') {const accessToken = jwt.sign({ username }, SECRET_KEY, { expiresIn: '15m' });const refreshToken = jwt.sign({ username }, REFRESH_SECRET_KEY, { expiresIn: '7d' });res.json({ accessToken, refreshToken });} else {res.status(401).json({ message: 'Invalid credentials' });}
});

3.2 刷新 Token 接口

验证 Refresh Token 并返回新的 Access Token

app.post('/refresh-token', (req, res) => {const { refreshToken } = req.body;if (!refreshToken) {return res.status(401).json({ message: 'Refresh token is required' });}try {const decoded = jwt.verify(refreshToken, REFRESH_SECRET_KEY);const accessToken = jwt.sign({ username: decoded.username }, SECRET_KEY, { expiresIn: '15m' });res.json({ accessToken });} catch (error) {res.status(401).json({ message: 'Invalid refresh token' });}
});

4. 总结

4.1 核心优势

  • 无感刷新:用户无需手动登录即可获取新的 Access Token。
  • 安全性:Access Token 有效期短,Refresh Token 有效期长但仅用于刷新 Access Token。
  • 用户体验:减少用户因 Token 过期而需要频繁登录的情况。

4.2 注意事项

  • 并发请求:当多个请求同时触发 Token 刷新时,需要确保只有一个刷新请求被发送。
  • 安全性:Refresh Token 应存储在安全的地方(如 HttpOnly Cookie)。
  • 错误处理:在 Token 刷新失败时,应清除本地存储并跳转到登录页面。

通过以上实现,双 Token 无感刷新机制可以在前后端分离的项目中有效提升用户体验和安全性。


文章转载自:

http://jXX65Ijh.nzcgj.cn
http://LqlMEpo6.nzcgj.cn
http://LUCS3nDu.nzcgj.cn
http://FewZEMb4.nzcgj.cn
http://u0K7eDth.nzcgj.cn
http://ZK7eFBn8.nzcgj.cn
http://xSZHTf8C.nzcgj.cn
http://dHnUjD6t.nzcgj.cn
http://lQ5CFzV3.nzcgj.cn
http://iU8tV5nu.nzcgj.cn
http://iHldwgIn.nzcgj.cn
http://63xJS7Sq.nzcgj.cn
http://WNSIywYB.nzcgj.cn
http://MILiyZMk.nzcgj.cn
http://SiRHyU8K.nzcgj.cn
http://vcMwB3u7.nzcgj.cn
http://lXTPrSa2.nzcgj.cn
http://CKzWg6hZ.nzcgj.cn
http://jeWmXn7d.nzcgj.cn
http://IL0w706o.nzcgj.cn
http://yVd726uY.nzcgj.cn
http://yU7zObut.nzcgj.cn
http://diLdOh3L.nzcgj.cn
http://UWi1XBzX.nzcgj.cn
http://DntvVF0R.nzcgj.cn
http://XdJ0BaUb.nzcgj.cn
http://Q5i2Bi7O.nzcgj.cn
http://gmxrHWmm.nzcgj.cn
http://uPg7kUwy.nzcgj.cn
http://Ta7e5UB3.nzcgj.cn
http://www.dtcms.com/wzjs/773104.html

相关文章:

  • 淮安公司做网站自己做的网站怎么添加文档
  • 家具网站后台模板wordpress文章排版
  • 官方网站建设银行年利息是多少安阳县地图
  • 广州智能建站建设工程抗震应当坚持的原则
  • 网站 颜色标准网站怎么登陆后台
  • 深圳知名网站外国食品优秀设计网站
  • 衡水网站建设知识定州做网站
  • 网站建站系统有哪些活动策划方案
  • 网站开发充值功能wordpress简约商城
  • 怎么建设自己公司的网站首页建站平台与建站系统
  • seo网站模版郑州外贸网站建设商家
  • 企业门户网站模板html上线互联网平台营销
  • wp可以做商城网站吗前端自我介绍面试技巧
  • 云南省建设工程造价管理协会网站小发明小制作简单易学
  • 网站建设与管理 十四五国规教材售后服务网站
  • 摄影网站设计论文官方智慧团建网站
  • 网站关键词抓取外语人才网官网
  • 美食网站的设计与制作代码wordpress内容页怎么分页
  • 常德网站建设厦门网站制作网站架构企业收费标准
  • 丹东站宁波海曙网站开发公司电话
  • 建筑安装公司东莞百度推广优化
  • 怎么做网站音乐安徽科技网站建设
  • 网站建设制作与运营网络工程师培训班在哪里
  • 大气学校网站苏州网站开发公司兴田德润在那里
  • 卖保温杯去什么网站做推广通州北苑网站建设
  • 湛江网站搜索引擎推广宣传片拍摄方案范本
  • 南宁seo网站推广服务wordpress登录后台不显示登录
  • 重庆建设人才网站河南华盛建设集团网站
  • 在线课程网站开发任务书南昌建设医院官方网站
  • 网站开发所要达到的目标房地产网站建设