亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

雙token無感刷新nodejs+React詳細解釋(保姆級教程)

 更新時間:2025年05月27日 09:42:58   作者:妄春山_1  
雙token系統(tǒng)可以更好地管理用戶的權(quán)限,這篇文章主要介紹了雙token無感刷新nodejs+React的相關(guān)資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下

雙token無感刷新是一種在Web開發(fā)中常用的安全及用戶體驗優(yōu)化技術(shù)。以下是對雙token無感刷新的詳細解釋:

一、基本概念

雙token無感刷新主要涉及兩種類型的token:Access Token(訪問令牌)和Refresh Token(刷新令牌)。

  • Access Token:用戶直接用于訪問受保護資源的憑證。有效期較短,通常幾分鐘到幾小時不等。一旦過期,用戶需要重新認證以獲取新的Access Token。即使Access Token被泄露,由于其有效期短,攻擊者利用它進行不當操作的時間窗口有限。
  • Refresh Token:用于在Access Token過期后或過期前的一定時間內(nèi)重新獲取新的Access Token。有效期通常較長,甚至可以說是永久的,但出于安全考慮,一般會設(shè)置過期時間或使用次數(shù)限制。Refresh Token通常不會直接發(fā)送給客戶端,而是保存在服務(wù)器端或經(jīng)過加密后存儲在客戶端本地(如localStorage或sessionStorage)。

二、工作原理

雙token無感刷新的核心在于自動在后臺處理Access Token的過期和刷新過程,而無需用戶重新登錄或感知到這一過程。具體流程如下:

  • 用戶通過用戶名(賬號)、密碼或其他認證方式向認證服務(wù)器請求授權(quán)。
  • 認證成功后,服務(wù)器返回Access Token和Refresh Token給前端。前端將這兩個Token保存到本地存儲中,以便在需要時使用。
  • 前端在訪問受保護資源時,將Access Token放入請求頭中發(fā)送給后端。
  • 如果Access Token有效,后端正常處理請求并返回結(jié)果。
  • 如果Access Token過期,后端會返回一個錯誤響應(如HTTP 401 Unauthorized)。
  • 前端在接收到錯誤響應后,自動在后臺使用Refresh Token向認證服務(wù)器請求新的Access Token。
  • 認證服務(wù)器驗證Refresh Token的有效性后,返回一個新的Access Token和(可選的)新的Refresh Token。
  • 前端更新本地存儲的Access Token和Refresh Token,并重新發(fā)起之前的請求。

三、實現(xiàn)方式

雙token無感刷新的實現(xiàn)通常依賴于前端和后端的配合。以下是一個簡化的實現(xiàn)流程:

  • 前端實現(xiàn)

    • 設(shè)置全局請求攔截器,在發(fā)送請求前檢查Access Token的有效性。
    • 如果Access Token即將過期或已過期,自動使用Refresh Token請求新的Access Token。
    • 更新本地存儲中的Access Token和Refresh Token。
    • 重新發(fā)起因Access Token過期而失敗的請求。
  • 后端實現(xiàn)

    • 提供一個用于刷新Token的接口,該接口接收Refresh Token作為參數(shù)。
    • 驗證Refresh Token的有效性,如果有效,則生成新的Access Token和(可選的)新的Refresh Token,并返回給客戶端。
    • 在受保護資源的訪問接口中驗證Access Token的有效性,并在其過期時返回相應的錯誤響應。

四、優(yōu)勢與應用場景

雙token無感刷新的優(yōu)勢在于提高了系統(tǒng)的安全性和用戶體驗。通過分離短期和長期的憑證,降低了Access Token被泄露的風險,同時避免了用戶因Token過期而頻繁重新登錄的麻煩。

這一機制廣泛應用于需要高安全性和良好用戶體驗的場景,如Web應用和移動應用。在這些場景中,雙token無感刷新能夠提升用戶在持續(xù)操作過程中的連貫性和安全性。

五、后端代碼解析(nodejs+express)

(1)app.js引入所需要的模塊 

//app.js
var createError = require('http-errors'); //用于創(chuàng)建各種HTTP錯誤
var express = require('express'); //引入express模塊
var path = require('path'); //用于處理文件和目錄的路徑
var cookieParser = require('cookie-parser'); //解析HTTP請求中的Cookie頭
var logger = require('morgan'); //用于將請求日志輸出到stdout或指定文件
var indexRouter = require('./routes/index'); //主路由
var usersRouter = require('./routes/users'); //用戶相關(guān)路由
var jwt = require("jsonwebtoken"); //引入jsonwebtoken模塊
var app = express();
var cors = require("cors")
app.use(cors()) //處理跨域

(2)自定義中間件,用于處理jwt驗證

//app.js
app.use((req, res, next) => {
  // 定義不需要驗證的路徑
  let pathArr = [
    '/userLogin',
    '/refresh',
    '/sendYzm',
    '/resetPassword',
    '/findUser',
    '/upload'
  ]

  // 如果請求路徑在不需要驗證的路徑中,直接調(diào)用next()繼續(xù)處理
  if (pathArr.includes(req.path)) {
    return next()
  }

  // 獲取請求頭中的accessToken和refreshToken
  const accessToken = req.headers.accesstoken
  const refreshToken = req.headers.refreshtoken


  // 判斷refreshToken是否過期
  try {
    jwt.verify(refreshToken, "MTHGH")
  } catch (error) {
    console.log(error);
    return res.status(403).send({ message: 'refreshToken驗證失敗' })
  }

  // 如果沒有accessToken 返回401
  if (!accessToken) {
    return res.status(401).send({ message: '未獲取到accessToken' })
  }
  // 驗證accessToken
  try {
    const user = jwt.verify(accessToken, "MTHGH")
    res.locals.user = user//將用戶信息存儲在res.locals中,供后續(xù)中間件使用
    return next()
  } catch (error) {
    return res.status(401).send({ message: 'accessToken驗證失敗' })
  }
})

(3)index.js所需引入模塊

// index.js
var express = require('express'); //引入了Express模塊
var router = express.Router(); //用于處理HTTP請求
const jwt = require('jsonwebtoken');  //實現(xiàn)用戶身份驗證和授權(quán)等功能

(4)封裝生成長token和短token的函數(shù)

// index.js
// 訪問令牌有效期
const ACCESS_TOKEN_EXPIRATION='1h'
// 刷新令牌有效期
const REFRESH_TOKEN_EXPIRATION='1d'
// 令牌密鑰
const SECRET_KEY="MTHGH"
// 這里創(chuàng)建了一個Map對象,用于存儲用戶名和對應的刷新令牌列表。這樣,當用戶登錄或刷新令牌時,可以跟蹤和驗證他們的刷新令牌。
const refreshTokenMap=new Map()

// 生成函數(shù)令牌
function generateToken(name,expiration){
  return jwt.sign({name},SECRET_KEY,{expiresIn:expiration})
}

// 封裝生成長token和短token
function getToken(name){
  let accessToken = generateToken(name,ACCESS_TOKEN_EXPIRATION)
  let refreshToken = generateToken(name,REFRESH_TOKEN_EXPIRATION)
  const refreshTokens = refreshTokenMap.get(name)||[]
  refreshTokens.push(refreshToken)
  refreshTokenMap.set(name,refreshTokens)
  return {
    accessToken,
    refreshToken
  }
}

(5)賬號密碼登錄

// index.js
router.post('/userLogin', async (req, res) => {
  // 獲取前端傳遞的數(shù)據(jù)
  const {username,password} = req.body
  // 通過用戶查找是否存在
  let user = await userModel.findOne({userName:username})
  // 不存在
  if(!user){
    return res.status(200).send({message:"賬號錯誤",code:1})
  }
  // 用戶存在,密碼不正確
  if(user.passWord!==password){
    return res.status(200).send({message:"密碼錯誤",code:2})
  }
  // 調(diào)用函數(shù),生成token
  let {accessToken,refreshToken}=getToken(user.userName)
  // 返回用戶、token
  res.status(200).send({
    data:user,
    accessToken,
    refreshToken,
    message:'登陸成功',
    code:200
  })
})

(6)刷新短token

// index.js
router.get('/refresh',async(req,res)=>{
  // 獲取請求頭的token信息
  const refreshToken = req.headers.refreshtoken
  // token不存在
  if(!refreshToken){
    res.status(403).send("沒有短token")
  }
  // 驗證token是否過期
  try{
    const {name} = jwt.verify(refreshToken,SECRET_KEY)
    const accessToken = generateToken(name,ACCESS_TOKEN_EXPIRATION)
    res.status(200).send({accessToken})
  }catch(error){
    console.log('長token已經(jīng)過期');
    res.status(403).send('長token已經(jīng)過期')
  }
})

六、前端代碼解析(React)

(1)添加axios重試機制

// api.jsx
let retryCount = 0; // 初始化重試計數(shù)
const customRetryCondition = async (error) => {
  // 自定義重試條件
  if (axios.isAxiosError(error) && error.response?.status !== 200) {
    // 如果是 Axios 錯誤且響應狀態(tài)不是 200
    if (error.response?.status === 403) {
      // 如果后端返回 403(禁止訪問)
      localStorage.removeItem('accessToken'); // 移除 accessToken
      localStorage.removeItem('refreshToken'); // 移除 refreshToken
      console.log('請重新登錄'); // 打印提示信息
      window.location.href='/login' // 跳轉(zhuǎn)到登錄頁面
      return false; // 不重試
    }

    if (error.response?.status === 401) {
      // 如果后端返回 401(未授權(quán))
      await refresh(); // 嘗試刷新 token
      console.log('刷新token'); // 打印提示信息
      return true; // 允許重試
    }

    retryCount++; // 增加重試計數(shù)
    console.log(`第${retryCount}次重試`); // 打印當前重試次數(shù)
    return (
      error.response.status >= 500 || // 如果響應狀態(tài)是 500 或以上
      (error.response.status < 500 && error.response?.status !== 401) // 或者狀態(tài)小于 500 但不等于 401
    );
  }
  return false; // 如果不符合條件,則不重試
};
// 配置 axios 實例的重試機制
axiosRetry(instance, {
  retries: 3, // 設(shè)置最多重試次數(shù)為 3 次
  retryCondition: customRetryCondition, // 使用自定義的重試條件
  retryDelay: axiosRetry.exponentialDelay, // 使用指數(shù)退避算法設(shè)置重試延遲
});

(2)axios添加請求攔截器

// api.jsx
// 請求攔截器
instance.interceptors.request.use(
  async function (config) {
    console.log('開始請求'); // 打印請求開始信息
    const accessToken = localStorage.getItem('accessToken'); // 從 localStorage 獲取 accessToken
    const refreshToken = localStorage.getItem('refreshToken'); // 從 localStorage 獲取 refreshToken
    console.log(accessToken);
    console.log(refreshToken);
    
    
    config.headers['accessToken'] = accessToken // 設(shè)置請求頭中的 accessToken
    config.headers['refreshToken'] = refreshToken // 設(shè)置請求頭中的 refreshToken
    return config; // 返回配置
  },
  function (error) {
    return Promise.reject(error); // 拒絕請求錯誤
  }
);

(3)axios添加響應攔截器

// api.jsx
/**
 * 響應攔截器
 */
instance.interceptors.response.use(
  async function (response) {
    // alert(222)
    if (response.status === 200) {
      
      return response; // 如果響應狀態(tài)是 200,返回響應
    } else {
      return Promise.reject(response.data.message || '未知錯誤'); // 否則拒絕并返回錯誤信息
    }
  },
  function (error) {
    if (error && error.response) {
      // 如果有響應錯誤
      switch (error.response.status) {
        case 400:
          error.message = '錯誤請求'; // 處理 400 錯誤
          break;
        case 401:
          error.message = '未授權(quán),請重新登錄'; // 處理 401 錯誤
          break;
        case 403:
          error.message = '拒絕訪問'; // 處理 403 錯誤
          localStorage.removeItem('accessToken'); // 移除 accessToken
          localStorage.removeItem('refreshToken'); // 移除 refreshToken
          
          // Router.push('/login'); // 跳轉(zhuǎn)到登錄頁面
          window.location.href='/login'
          break;
        case 404:
          error.message = '請求錯誤,未找到該資源'; // 處理 404 錯誤
          break;
        case 405:
          error.message = '請求方法未允許'; // 處理 405 錯誤
          break;
        case 408:
          error.message = '請求超時'; // 處理 408 錯誤
          break;
        case 500:
          error.message = '服務(wù)器端出錯'; // 處理 500 錯誤
          break;
        case 501:
          error.message = '網(wǎng)絡(luò)未實現(xiàn)'; // 處理 501 錯誤
          break;
        case 502:
          error.message = '網(wǎng)絡(luò)錯誤'; // 處理 502 錯誤
          break;
        case 503:
          error.message = '服務(wù)不可用'; // 處理 503 錯誤
          break;
        case 504:
          error.message = '網(wǎng)絡(luò)超時'; // 處理 504 錯誤
          break;
        case 505:
          error.message = 'http版本不支持該請求'; // 處理 505 錯誤
          break;
        default:
          error.message = `連接錯誤${error.response.status}`; // 處理其他未知錯誤
      }
    } else {
      error.message = '連接服務(wù)器失敗'; // 如果沒有響應,打印連接失敗信息
    }
    return Promise.reject(error.message); // 拒絕并返回錯誤信息
  }
);

(4)刷新token

// api.jsx
// 重新刷新token
 async function refresh() {
  let res =await instance.get('/refresh'); // 發(fā)送請求以刷新 token
  localStorage.setItem('accessToken', res.data.accessToken); // 將新的 accessToken 存儲到 localStorage
}

七、后端完整代碼

(1)app.js

// app.js

var createError = require('http-errors');
var express = require('express');
var path = require('path');
var cookieParser = require('cookie-parser');
var logger = require('morgan');

var indexRouter = require('./routes/index');
var usersRouter = require('./routes/users');
var jwt = require("jsonwebtoken");
var app = express();
// cors
var cors = require("cors")
app.use(cors())

// 自定義中間件 用于處理jwt驗證
app.use((req, res, next) => {
  // 定義不需要驗證的路徑
  let pathArr = [
    '/userLogin',
    '/refresh',
    '/sendYzm',
    '/resetPassword',
    '/findUser',
    '/upload'
  ]

  // 如果請求路徑在不需要驗證的路徑中,直接調(diào)用next()繼續(xù)處理
  if (pathArr.includes(req.path)) {
    return next()
  }

  // 獲取請求頭中的accessToken和refreshToken
  const accessToken = req.headers.accesstoken
  const refreshToken = req.headers.refreshtoken


  // 判斷refreshToken是否過期
  try {
    jwt.verify(refreshToken, "MTHGH")
  } catch (error) {
    console.log(error, 111);
    return res.status(403).send({ message: 'refreshToken驗證失敗' })
  }

  // 如果沒有accessToken 返回401
  if (!accessToken) {
    return res.status(401).send({ message: '未獲取到accessToken' })
  }
  // 使用中間件
  // app.use('/index',validateRefreshToken,validateAccessToken)
  // app.use('/user',validateRefreshToken,validateAccessToken)
  // 驗證accessToken
  try {
    const user = jwt.verify(accessToken, "MTHGH")
    res.locals.user = user//將用戶信息存儲在res.locals中,供后續(xù)中間件使用
    return next()
  } catch (error) {
    return res.status(401).send({ message: 'accessToken驗證失敗' })
  }
})


// token
// var expressJWT = require("express-jwt")
// app.use(expressJWT({
//     secret: "123",
//     algorithms: ["HS256"]
// }).unless({
//     path: ["/login", "/upload", {url: /^\/upload/, methods: ['GET']}]
// }))

// view engine setup
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'jade');

app.use(logger('dev'));
app.use(express.json());
app.use(express.urlencoded({ extended: false }));
app.use(cookieParser());
app.use(express.static(path.join(__dirname, 'public')));
app.use('/upload', express.static(path.join(__dirname, 'upload')));

app.use('/', indexRouter);
app.use('/users', usersRouter);

// catch 404 and forward to error handler
app.use(function (req, res, next) {
  next(createError(404));
});

// error handler
app.use(function (err, req, res, next) {
  // set locals, only providing error in development
  res.locals.message = err.message;
  res.locals.error = req.app.get('env') === 'development' ? err : {};

  // render the error page
  res.status(err.status || 500);
  res.render('error');
});

module.exports = app;

(2)index.js

// index.js
var express = require('express');
var router = express.Router();
const jwt = require('jsonwebtoken');

var { userModel } = require("../model/model")
const app = express();
app.use(express.json()); // 解析請求中的 JSON 數(shù)據(jù)

// 訪問令牌有效期
const ACCESS_TOKEN_EXPIRATION='1h'
// 刷新令牌有效期
const REFRESH_TOKEN_EXPIRATION='1d'
const SECRET_KEY="MTHGH"
const refreshTokenMap=new Map()

// 生成函數(shù)令牌
function generateToken(name,expiration){
  return jwt.sign({name},SECRET_KEY,{expiresIn:expiration})
}

// 封裝生成長token和短token
function getToken(name){
  let accessToken = generateToken(name,ACCESS_TOKEN_EXPIRATION)
  let refreshToken = generateToken(name,REFRESH_TOKEN_EXPIRATION)
  const refreshTokens = refreshTokenMap.get(name)||[]
  refreshTokens.push(refreshToken)
  refreshTokenMap.set(name,refreshTokens)
  return {
    accessToken,
    refreshToken
  }
}

// 賬號密碼登錄
router.post('/userLogin', async (req, res) => {
  const {username,password} = req.body
  let user = await userModel.findOne({userName:username})
  if(!user){
    return res.status(200).send({message:"賬號錯誤",code:1})
  }
  if(user.passWord!==password){
    return res.status(200).send({message:"密碼錯誤",code:2})
  }
  let {accessToken,refreshToken}=getToken(user.userName)
  res.status(200).send({
    data:user,
    accessToken,
    refreshToken,
    message:'登陸成功',
    code:200
  })
})

// 刷新短token
router.get('/refresh',async(req,res)=>{
  const refreshToken = req.headers.refreshtoken
  // console.log(111);
  
  if(!refreshToken){
    res.status(403).send("沒有短token")
  }
  try{
    const {name} = jwt.verify(refreshToken,SECRET_KEY)
    const accessToken = generateToken(name,ACCESS_TOKEN_EXPIRATION)
    res.status(200).send({accessToken})
  }catch(error){
    console.log('長token已經(jīng)過期');
    res.status(403).send('長token已經(jīng)過期')
    
  }
})

module.exports = router;

八、前端完整代碼

(1)api.jsx

// api.jsx
import axios from "axios";
import axiosRetry from 'axios-retry';

// 基礎(chǔ)配置
const instance = axios.create({
  baseURL: "http://localhost:3010",
  timeout: 5000,
  headers: { 'Content-Type': 'application/json' },
});

/**
 * 重試機制
 */
let retryCount = 0; // 初始化重試計數(shù)
const customRetryCondition = async (error) => {
  // 自定義重試條件
  if (axios.isAxiosError(error) && error.response?.status !== 200) {
    // 如果是 Axios 錯誤且響應狀態(tài)不是 200
    if (error.response?.status === 403) {
      // 如果后端返回 403(禁止訪問)
      localStorage.removeItem('accessToken'); // 移除 accessToken
      localStorage.removeItem('refreshToken'); // 移除 refreshToken
      console.log('請重新登錄'); // 打印提示信息
      window.location.href='/login' // 跳轉(zhuǎn)到登錄頁面
      return false; // 不重試
    }

    if (error.response?.status === 401) {
      // 如果后端返回 401(未授權(quán))
      await refresh(); // 嘗試刷新 token
      console.log('刷新token'); // 打印提示信息
      return true; // 允許重試
    }

    retryCount++; // 增加重試計數(shù)
    console.log(`第${retryCount}次重試`); // 打印當前重試次數(shù)
    return (
      error.response.status >= 500 || // 如果響應狀態(tài)是 500 或以上
      (error.response.status < 500 && error.response?.status !== 401) // 或者狀態(tài)小于 500 但不等于 401
    );
  }
  return false; // 如果不符合條件,則不重試
};

// 配置 axios 實例的重試機制
axiosRetry(instance, {
  retries: 3, // 設(shè)置最多重試次數(shù)為 3 次
  retryCondition: customRetryCondition, // 使用自定義的重試條件
  retryDelay: axiosRetry.exponentialDelay, // 使用指數(shù)退避算法設(shè)置重試延遲
});

// 請求攔截器
instance.interceptors.request.use(
  async function (config) {
    console.log('開始請求'); // 打印請求開始信息
    const accessToken = localStorage.getItem('accessToken'); // 從 localStorage 獲取 accessToken
    const refreshToken = localStorage.getItem('refreshToken'); // 從 localStorage 獲取 refreshToken
    console.log(accessToken);
    console.log(refreshToken);
    
    
    config.headers['accessToken'] = accessToken // 設(shè)置請求頭中的 accessToken
    config.headers['refreshToken'] = refreshToken // 設(shè)置請求頭中的 refreshToken
    return config; // 返回配置
  },
  function (error) {
    return Promise.reject(error); // 拒絕請求錯誤
  }
);

/**
 * 響應攔截器
 */
instance.interceptors.response.use(
  async function (response) {
    // alert(222)
    if (response.status === 200) {
      
      return response; // 如果響應狀態(tài)是 200,返回響應
    } else {
      return Promise.reject(response.data.message || '未知錯誤'); // 否則拒絕并返回錯誤信息
    }
  },
  function (error) {
    if (error && error.response) {
      // 如果有響應錯誤
      switch (error.response.status) {
        case 400:
          error.message = '錯誤請求'; // 處理 400 錯誤
          break;
        case 401:
          error.message = '未授權(quán),請重新登錄'; // 處理 401 錯誤
          break;
        case 403:
          error.message = '拒絕訪問'; // 處理 403 錯誤
          localStorage.removeItem('accessToken'); // 移除 accessToken
          localStorage.removeItem('refreshToken'); // 移除 refreshToken
          
          // Router.push('/login'); // 跳轉(zhuǎn)到登錄頁面
          window.location.href='/login'
          break;
        case 404:
          error.message = '請求錯誤,未找到該資源'; // 處理 404 錯誤
          break;
        case 405:
          error.message = '請求方法未允許'; // 處理 405 錯誤
          break;
        case 408:
          error.message = '請求超時'; // 處理 408 錯誤
          break;
        case 500:
          error.message = '服務(wù)器端出錯'; // 處理 500 錯誤
          break;
        case 501:
          error.message = '網(wǎng)絡(luò)未實現(xiàn)'; // 處理 501 錯誤
          break;
        case 502:
          error.message = '網(wǎng)絡(luò)錯誤'; // 處理 502 錯誤
          break;
        case 503:
          error.message = '服務(wù)不可用'; // 處理 503 錯誤
          break;
        case 504:
          error.message = '網(wǎng)絡(luò)超時'; // 處理 504 錯誤
          break;
        case 505:
          error.message = 'http版本不支持該請求'; // 處理 505 錯誤
          break;
        default:
          error.message = `連接錯誤${error.response.status}`; // 處理其他未知錯誤
      }
    } else {
      error.message = '連接服務(wù)器失敗'; // 如果沒有響應,打印連接失敗信息
    }
    return Promise.reject(error.message); // 拒絕并返回錯誤信息
  }
);

// 重新刷新token
 async function refresh() {
  let res =await instance.get('/refresh'); // 發(fā)送請求以刷新 token
  localStorage.setItem('accessToken', res.data.accessToken); // 將新的 accessToken 存儲到 localStorage
}

// 導出封裝后的 axios 實例
export default instance;

綜上所述,雙token無感刷新是一種有效的安全及用戶體驗優(yōu)化技術(shù),通過合理的Token管理和自動刷新機制,實現(xiàn)了在不影響用戶體驗的前提下提升系統(tǒng)安全性的目標。

總結(jié)

到此這篇關(guān)于雙token無感刷新nodejs+React的文章就介紹到這了,更多相關(guān)雙token無感刷新nodejs+React內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • async/await優(yōu)雅的錯誤處理方法總結(jié)

    async/await優(yōu)雅的錯誤處理方法總結(jié)

    這篇文章主要給大家介紹了關(guān)于async/await優(yōu)雅的錯誤處理方法的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-01-01
  • Node.js Addons翻譯(C/C++擴展)

    Node.js Addons翻譯(C/C++擴展)

    這篇文章主要介紹了Node.js Addons翻譯(C/C++擴展) 的相關(guān)資料,非常不錯具有參考借鑒價值,需要的朋友可以參考下
    2016-06-06
  • 教你徹底搞懂ESM與CJS互相轉(zhuǎn)換

    教你徹底搞懂ESM與CJS互相轉(zhuǎn)換

    這篇文章主要為大家介紹了ESM與CJS互相轉(zhuǎn)換的理解與實現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-03-03
  • nodejs利用http模塊實現(xiàn)銀行卡所屬銀行查詢和騷擾電話驗證示例

    nodejs利用http模塊實現(xiàn)銀行卡所屬銀行查詢和騷擾電話驗證示例

    本篇文章主要介紹了nodejs利用http模塊實現(xiàn)銀行卡所屬銀行查詢和騷擾電話驗證示例,有興趣的可以了解一下。
    2016-12-12
  • 詳解npm和cnpm混用的坑

    詳解npm和cnpm混用的坑

    有沒有遇到過npm和cnpm一起用的時候出現(xiàn)奇奇怪怪的問題呢? 有沒有遇到過cnpm在支付寶小程序上面安裝包無效?本文就詳解一下npm和cnpm混用的坑,感興趣的可以了解下
    2021-07-07
  • node+express+ejs使用模版引擎做的一個示例demo

    node+express+ejs使用模版引擎做的一個示例demo

    本篇文章主要介紹了node+express+ejs使用模版引擎做的一個示例demo,具有一定參考價值,有興趣的小伙伴可以了解一下
    2017-09-09
  • Egret引擎開發(fā)指南之編譯項目

    Egret引擎開發(fā)指南之編譯項目

    Egret框架是一個基于MIT開源協(xié)議許可的永久免費的項目!你可以在項目中隨意使用且修改它,并且擁有100%的控制權(quán)。你可以從Egret的Github網(wǎng)站獲取它的源代碼,從而了解和學習它的核心細節(jié)。Egret具有完善的文檔,并且易于上手學習,可以讓你更容易專注于游戲本身的開發(fā)
    2014-09-09
  • 基于Node.js + WebSocket打造即時聊天程序嗨聊

    基于Node.js + WebSocket打造即時聊天程序嗨聊

    這篇文章主要介紹了基于Node.js + WebSocket打造即時聊天程序,有興趣的可以了解一下。
    2016-11-11
  • 修改NPM全局模式的默認安裝路徑的方法

    修改NPM全局模式的默認安裝路徑的方法

    這篇文章主要介紹了修改NPM全局模式的默認安裝路徑的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-12-12
  • Node.js定時任務(wù)之node-schedule使用詳解

    Node.js定時任務(wù)之node-schedule使用詳解

    這篇文章主要介紹了Node.js定時任務(wù)之node-schedule使用詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08

最新評論