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

怎么做网站写书/品牌网络推广怎么做

怎么做网站写书,品牌网络推广怎么做,页面设计要会什么,wordpress gzip 和 cache 冲突吗Node.js中JWT的token完整生命周期管理:从生成到销毁 在Node.js中使用JWT(JSON Web Token)进行身份验证和授权是一种常见的实践。下面详细介绍JWT从生成到销毁的过程。 JWT生成 安装jsonwebtoken库: 要生成JWT,首先…

Node.js中JWT的token完整生命周期管理:从生成到销毁

在Node.js中使用JWT(JSON Web Token)进行身份验证和授权是一种常见的实践。下面详细介绍JWT从生成到销毁的过程。

JWT生成

安装jsonwebtoken库

要生成JWT,首先需要安装jsonwebtoken库。通过运行以下命令来安装:

npm install jsonwebtoken

创建JWT

在Node.js中生成JWT(JSON Web Token)的过程涉及几个关键步骤,下面将详细介绍这些步骤和相关的技术细节。

1. Header(头部)

JWT的头部通常由两部分组成:令牌的类型(typ)和所使用的签名算法(alg)。例如,对于HS256算法,头部可能如下所示:

{"alg": "HS256","typ": "JWT"
}

这个JSON对象需要被Base64Url编码,以形成JWT的第一个部分

2. Payload(负载)

负载部分包含了所谓的“声明”(Claims),这些是关于实体(通常是用户)和其他数据的陈述。声明分为三类:注册的声明、公共的声明和私有的声明。例如:

{"sub": "1234567890","name": "John Doe","admin": true
}

这个JSON对象同样需要被Base64Url编码,以形成JWT的第二部分

3. Signature(签名)

签名是使用Header中指定的签名算法和密钥(Secret)对Header和Payload的编码字符串进行签名的结果。这个过程确保了JWT的完整性和真实性。签名的生成公式如下:

HMACSHA256(base64UrlEncode(header) + "." +base64UrlEncode(payload),secret)

这个签名是JWT的第三部分,并且是验证JWT有效性的关键。

4. 组合Header、Payload和Signature

最终,将编码后的Header、Payload和生成的Signature用点(.)连接起来,形成完整的JWT字符串

eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c
5. 使用jsonwebtoken库生成JWT

在Node.js中,可以使用jsonwebtoken库来简化JWT的生成过程。以下是一个基本示例:

const jwt = require('jsonwebtoken');
const token = jwt.sign({id: 1,username: 'admin'},'secret', // 自定义密钥{expiresIn: '1h' // 过期时间}
);
console.log(token);

在这个示例中,我们使用jwt.sign方法,传入负载(包含用户信息)、密钥和签名选项(如过期时间)来生成JWT。

通过这些步骤,你可以在Node.js应用中生成JWT,用于身份验证和信息交换。生成的JWT可以被客户端存储和使用,直到它过期或被服务器端撤销。

JWT验证

在Node.js中验证JWT的过程涉及几个关键步骤,以下是详细的验证方法:

1. 验证JWT

服务器收到JWT后,需要验证其真实性。使用以下代码验证JWT:

const jwt = require('jsonwebtoken');
const token = 'your-jwt-token';
const secretKey = 'your-secret-key';try {const decoded = jwt.verify(token, secretKey);console.log(decoded);
} catch (error) {console.error('JWT verification failed');
}

在这个过程中,jwt.verify方法会使用提供的secretKey来验证JWT的签名。如果签名验证成功,它会返回JWT的Payload部分,否则会抛出错误。

2. 创建JWT验证中间件

在Express.js中,你可以创建JWT验证中间件来保护特定路由。以下是一个示例:

const jwt = require('jsonwebtoken');
const secretKey = 'your-secret-key';function authenticateToken(req, res, next) {const token = req.header('Authorization');if (!token) return res.status(401).send('Access denied');try {const decoded = jwt.verify(token, secretKey);req.user = decoded;next();} catch (error) {res.status(403).send('Invalid token');}
}

这个中间件函数authenticateToken会检查HTTP请求头中的Authorization字段是否包含JWT。如果JWT存在,它会尝试验证JWT的有效性。如果验证成功,decoded对象(包含JWT的Payload)会被附加到req.user上,以便在后续的请求处理中使用。如果验证失败,会返回403错误。

3. 应用中间件保护路由

你可以将authenticateToken中间件应用到需要保护的路由上,以确保只有验证通过的用户才能访问:

app.get('/protected', authenticateToken, (req, res) => {res.json({ message: 'This is a protected route', user: req.user });
});

在这个例子中,任何尝试访问/protected路由的请求都必须通过authenticateToken中间件的验证。

4. 安全提示

  • 安全存储密钥:密钥是JWT的关键部分,应安全存储。不要硬编码密钥,最好从环境变量或配置文件中读取。
  • 定期刷新令牌:为JWT设置适当的到期时间,以降低安全风险。客户端应定期获取新的JWT令牌。
  • 不要在JWT中存储敏感信息:避免在JWT中存储敏感信息,因为JWT可以被解码。如果需要存储敏感信息,应使用加密而不是签名。

通过这些步骤,你可以在Node.js应用中实现JWT的验证,确保只有持有有效JWT的用户才能访问受限资源。

JWT销毁

客户端销毁

客户端通常负责销毁存储在本地的JWT,例如在浏览器中清除localStorage或sessionStorage中的token,但是这种方法通常不保险,还需要在后端彻底将推出的jwt销毁。这可以通过设置cookie过期时间来实现:

res.cookie('token', 'none', {expires: new Date(Date.now()),
})

服务端销毁

由于只在服务端删除浏览器中的localStorage不保险,所以子在服务端也要进行类似的操作,将已经生成的token删除掉。结合本人已经做过的项目代码进行一步步讲解:

1.创建一个tokenmanage表

这个表用来存储每一个登录的账号的登录时间(请求登录接口时候的登录时间)、过期时间(你token设置的过期时间)、用户账号(生成这个token对应的账号,对应用户表的account)

SET FOREIGN_KEY_CHECKS=0;-- ----------------------------
-- Table structure for `tokenmanage`
-- ----------------------------
DROP TABLE IF EXISTS `tokenmanage`;
CREATE TABLE `tokenmanage` (`id` int(8) NOT NULL AUTO_INCREMENT,`token` varchar(255) DEFAULT NULL,`logintime` varchar(255) DEFAULT NULL COMMENT '登录时间',`expiretime` varchar(255) DEFAULT NULL COMMENT '过期时间',`useraccount` varchar(255) DEFAULT NULL COMMENT '用户账号',PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf32;
2. user表加字段
SET FOREIGN_KEY_CHECKS=0;-- ----------------------------
-- Table structure for `users`
-- ----------------------------
DROP TABLE IF EXISTS `users`;
CREATE TABLE `users` (`id` int(8) NOT NULL AUTO_INCREMENT,`account` varchar(30) DEFAULT NULL,`pwd` varchar(30) DEFAULT NULL,`utype` int(1) DEFAULT NULL COMMENT '类型',`status` tinyint(1) DEFAULT '1' COMMENT '状态:1 正常  0删除',`ctime` varchar(30) DEFAULT NULL COMMENT '创建时间,时间戳',`pfcollege` varchar(30) DEFAULT NULL,`pfgrade` varchar(10) DEFAULT NULL,`pfspeciality` varchar(10) DEFAULT NULL,`pfpart` tinyint(1) DEFAULT NULL,`uname` varchar(10) DEFAULT NULL,`picture` varchar(100) DEFAULT NULL,`logintime` varchar(30) DEFAULT NULL COMMENT '登录时间', PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=36 DEFAULT CHARSET=utf32;
2.登录接口处理

所需模块:

var express = require('express')
var router = express.Router()
var mysql = require('mysql')
// 生成token
const webjwt = require('jsonwebtoken')//  从db文件中导入的数据库模块
var mysql = require("../db/mysql")

登录接口代码:

// 登录接口
router.get('/login', function (req, response) {var body = req.query;console.log(req.query);let { mobile, pwd } = body;console.log(mobile, pwd);// 先查询账号是否存在mysql.query('SELECT account, id, logintime FROM users WHERE account = ?', [mobile], function (error, results) {if (error) {console.log(error);response.json({ code: 1, msg: '数据库查询错误' });return;}if (results.length > 0) {// 如果账号存在的话继续查询账号密码mysql.query('SELECT account, id FROM users WHERE account = ? AND pwd = ?', [mobile, pwd], function (error, results) {if (error) {console.log(error);response.json({ code: 1, msg: '数据库查询错误' });return;}if (results.length > 0) {let user = results[0];// 更新登录时间let now = new Date().getTime(); // 获取当前时间的时间戳,以毫秒为单位mysql.query('UPDATE users SET logintime = ? WHERE id = ?', [now, user.id], function (error, updateResults) {if (error) {console.log(error);response.json({ code: 1, msg: '数据库更新错误' });return;}// 注意默认情况 Token 必须以 Bearer+空格 开头const token = 'Bearer ' + webjwt.sign({uid: user.id,account: user.account},'pwdxpx',{expiresIn: 3600 * 24 * 2// expiresIn: 1 * 60});let threeDaysLater = now + 2 * 24 * 60 * 60 * 1000; // 计算三天后的时间戳// let threeDaysLater = now + 1 * 60 * 1000;// 检查tokenManage表中是否存在对应的useraccount记录,看看这个用户是否登录过mysql.query('SELECT token,logintime,expiretime FROM tokenmanage WHERE useraccount = ?',[user.account],function (error, tokenResults) {console.log(tokenResults[0].logintime, typeof (tokenResults[0].expiretime))if (error) {console.log(error);response.json({ code: 1, msg: '数据库查询错误' });return;}if (tokenResults.length > 0) {// 如果当前账号存在token,那就判断登录时间是否超过了过期时间,// 如果登录token已经过期了,那就删除当前记录if (Number(tokenResults[0].logintime) > Number(tokenResults[0].expiretime)) {// 删除+插入添加/* mysql.query("DELETE FROM tokenmanage WHERE useraccount = ?", [user.account], (error, resultsexpire) => {if (error) console.log(error)console.log("token 验证已经过期了,接下来重新添加")})mysql.query('INSERT INTO tokenmanage (token, logintime, expiretime, useraccount) VALUES (?, ?, ?, ?)',[token, now, threeDaysLater, user.account],function (error, insertTokenResults) {if (error) {console.log(error);} else {console.log("token表添加成功!");}});*/// 直接更新 ~~删除+插入添加mysql.query("UPDATE tokenmanage SET token = ?, logintime = ?, expiretime = ?, useraccount = ?",[token, now, threeDaysLater, user.account],(error, updatetokenResults) => {if (error) {console.log(error);} else {console.log("token表更新成功!");response.json({code: 0, msg: "验证更新-登录完成",data: {userid: user.id,user_account: mobile,status: 'ok',token: token}});}})} else {// 如果登录token没有过期了,那就更新当前记录的登录时间改为最新的时间mysql.query('UPDATE tokenmanage SET logintime = ? WHERE useraccount = ?',[now, user.account],function (error, updateTokenResults) {if (error) {console.log(error);} else {console.log("token表的登录时间更新成功!");}});response.json({code: 0, msg: "登录完成",data: {userid: user.id,user_account: mobile,status: 'ok',token: tokenResults[0].token}});}} else {// 如果当前账号不存在token,那就插入一条新记录mysql.query('INSERT INTO tokenManage (token, logintime, expiretime, useraccount) VALUES (?, ?, ?, ?)',[token, now, threeDaysLater, user.account],function (error, insertTokenResults) {if (error) {console.log(error);} else {console.log("token表添加成功!");}});// 账号密码一致则返回用户需要的数据response.json({code: 0, msg: "登录完成",data: {userid: user.id,user_account: mobile,status: 'ok',token: token}});}});});} else {response.json({code: 1,msg: '用户名或密码不正确',});}});} else {// 账号不存在则提示:账号不存在,请输入正确的账号response.send({ code: 1, msg: '账号不存在,请输入正确的账号' });}});
});
3. 退出接口处理
// 退出接口
router.get('/logout', function (req, res, next) {// 从请求头中获取Authorization字段,并提取tokenconst authHeader = req.headers.authorization;const token = authHeader ? authHeader.split(' ')[1] : null; // 提取tokenif (!token) {return res.status(401).json({ msg: "未提供有效的token", code: 401 });}console.log(token);// 先通过请求头中提供的token,去数据表中根据token字段查找是否存在该记录let sqltoken = `SELECT useraccount, logintime, expiretime FROM tokenmanage WHERE token = ?`;mysql.query(sqltoken, [token], (error, results) => {if (error) {console.log(error);return res.status(500).json({ msg: "服务器错误", code: 500 });} else {// 如果查找语句查出结果代表此token已经存在记录,要做到退出就将此token对应的记录删除即可if (results.length === 0) {return res.status(404).json({ msg: "未找到token", code: 404 });}// 假设你只想返回一个退出成功的消息,而不是整个results对象var msg = { "msg": "退出成功", code: 0 };// 删除tokenmanage表中对应的记录let deleteSql = `DELETE FROM tokenmanage WHERE token = ?`;mysql.query(deleteSql, [token], (deleteError, deleteResults) => {if (deleteError) {console.log(deleteError);return res.status(500).json({ msg: "删除token失败", code: 500 });}// 如果删除成功,返回成功消息res.json(msg);});}});
});

通过上述步骤,你可以在Node.js应用中实现JWT的生成、验证和销毁,从而确保应用的安全性和用户身份的验证。

相关文章:

Node.js中JWT的token完整生命周期管理:从生成到销毁

Node.js中JWT的token完整生命周期管理:从生成到销毁 在Node.js中使用JWT(JSON Web Token)进行身份验证和授权是一种常见的实践。下面详细介绍JWT从生成到销毁的过程。 JWT生成 安装jsonwebtoken库: 要生成JWT,首先…...

Kotlin报错:lateinit property xxx has not been initialized

Kotlin报错:lateinit property xxx has not been initialized 发生在定义了一个名为xxx的lateinit变量。 解决,在调用前,可以先判断一层该xxx变量是否已经初始化: if (this::xxx.isInitialized) {//正常使用该变量} kotlin.Unini…...

debian编译失败

A、缘由和分析 debian的代码在删除该路径下的2个包后, 重新全编,编译不过的问题。 至于我为什么删除这2个包,这是因为在sdk第一次编译时一些文件已经打包进去了,我现在的修改无法更新进img中,而现在我的项目中不需要…...

flink-connector-mysql-cdc:03 mysql-cdc常见问题汇总

flink-connector-mysql-cdc: 01 mysql-cdc基础配置代码演示02 mysql-cdc高级扩展03 mysql-cdc常见问题汇总04 mysql-cdc-kafka生产级代码分享05 flink-kafka-doris生产级代码分享06 flink-kafka-hudi生产级代码分享flink-cdc版本:3.2.0 flink版本:flink-1.18.0 mysql版本:…...

JSP技术发展现状

多年前,Java入门时学习的JSP可谓时风光无限,J2EE如日中天,短短数年,技术迭代更新光速般发展,有些技术慢慢就退出历史舞台。 JSP(Java Server Pages) 技术在早期 Java Web 开发中曾是构建动态网…...

浏览器同源策略、跨域、跨域请求,服务器处理没、跨域解决方案

目录 什么是同源策略什么是跨域发生跨域时,服务器有没有接到请求并处理响应:(两种情况) 如何解决跨域 什么是同源策略 概念: 同源策略是浏览器的一种安全机制,用于防止恶意网站对用户的敏感数据进行未经授…...

flink-connector-mysql-cdc:02 mysql-cdc高级扩展

flink-connector-mysql-cdc:01 mysql-cdc基础配置代码演示02 mysql-cdc高级扩展03 mysql-cdc常见问题汇总04 mysql-cdc-kafka生产级代码分享05 flink-kafka-doris生产级代码分享06 flink-kafka-hudi生产级代码分享 flink-cdc版本:3.2.0flink版本&#xf…...

Couchbase 简介

Couchbase 是一款分布式 NoSQL 数据库,主要用于现代应用程序中高性能、高可扩展性和灵活的数据存储需求。它结合了文档存储和键值存储的特性,为开发者提供了一种高效的数据库解决方案。 Couchbase 的特点 高性能: 支持内存优先的架构&#x…...

我们来学mysql -- 事务并发之幻读(原理篇)

事务并发之幻读 题记幻读系列文章 题记 在《事务之概念》提到事务对应现实世界的状态转换,这个过程要满足4个特性这世界,真理只在大炮射程之类,通往和平的道路,非“常人”可以驾驭一个人生活按部就班,人多起来&#x…...

Ubuntu Linux 图形界面工具管理磁盘分区和文件系统(八)

本文为Ubuntu Linux操作系统- 第八弹~~ 今天接着上文的内容,讲Linux磁盘分区存储的相关知识~ 上期回顾:命令行-管理磁盘分区和文件系统 今天看酷酷的雪獒铠甲!!雪獒铠甲合体~ 文章目录 磁盘管理器GNOME Disks主要功能安装命令 磁盘…...

Eclipse IDE 各个版本的用途和区别

Eclipse官方下载地址:https://www.eclipse.org/downloads/packages/ 会出现很多个Eclipse版本,初学者可能会感觉到很迷惑,不知道下载哪个版本。 Eclipse IDE for Enterprise Java and Web Developers (544 MB) 专为 Java 和 Web 应用开发者设计 包含 Java IDE、JavaScript、…...

国产GPU中,VLLM0.5.0发布Qwen2.5-14B-Instruct-GPTQ-Int8模型,请求返回结果乱码

概述 国产GPU: DCU Z100 推理框架: vllm0.5.0 docker容器化部署 运行如下代码: python -m vllm.entrypoints.openai.api_server --model /app/models/Qwen2.5-14B-Instruct-GPTQ-Int8 --served-model-name qwen-gptq --trust-remote-code --enforce…...

在 Vue 3 中实现点击按钮后禁止浏览器前进或后退

在 Vue 3 中实现点击按钮后禁止浏览器前进或后退&#xff0c;我们可以通过 ref 和 watch 来管理状态&#xff0c;同时使用 onBeforeUnmount 来清理事件监听。 使用 Vue 3 实现&#xff1a; <template><div><button click"disableNavigation">点击…...

Linux:软硬链接

目录 一、概念 软链接 硬链接 二、原理 硬链接 软链接 三、使用场景 硬链接 软链接 一、概念 软链接 在当前目录下&#xff0c;有一个普通文件a.txt。 ln -s a.txt a_soft.link结论&#xff1a; 软链接是一个文件。 观察inode_id&#xff0c;发现软链接有着独立…...

Delphi XE 安卓Web开发 错误:net::ERR_CLEARTEXT_NOT_PERMITTED

解决方法&#xff1a; 1、确保已经申明权限&#xff08;AndroidManifest.xml 文件&#xff09; 1 <uses-permission android:name"android.permission.INTERNET" /> 2、开启 usesCleartextTraffic 1 2 <application android:usesCleartextTraffic&qu…...

深入理解malloc与vector:内存管理的对比

引言‌ 在编程中&#xff0c;内存管理是一个至关重要的环节。无论是C语言中的malloc函数&#xff0c;还是C标准库中的vector容器&#xff0c;它们都在内存分配和释放上扮演着关键角色。然而&#xff0c;它们的设计理念和用法有着显著的不同。本文将深入探讨malloc和vector的区…...

多个输入框联合搜索

如果你有多个输入框&#xff0c;并希望进行联合精准搜索&#xff0c;可以通过组合多个输入框的值来过滤数据。在JavaScript中&#xff0c;常见的做法是先收集每个输入框的值&#xff0c;然后使用这些值过滤数据。 示例&#xff1a;多个输入框联合精准搜索 假设有多个输入框用…...

笔记03----NeurIPS2024 涨点!SSA:用于语义分割的语义和空间自适应像素级分类器(即插即用)

前言 文章标题&#xff1a;《SSA-Seg: Semantic and Spatial Adaptive Pixel-level Classiffer for Semantic Segmentation》 助力语义分割涨点!SSA:一种新颖的语义和空间自适应分类器&#xff0c;显著提高了基线模型的分割性能&#xff0c;比如SegNeXt、OCRNet和UperNet等模型…...

自定义比较函数 down 作为 sort 函数的参数实现数组元素从大到小排序

【自定义比较函数 down 作为 sort 函数的参数实现数组元素从大到小排序】 #include <bits/stdc.h> using namespace std;const int maxn1e35; int a[maxn];bool down(int u,int v) {return u>v; }int main() {int n;cin>>n;for(int i0; i<n; i) cin>>…...

在 Spring Boot 中使用 JPA(Java Persistence API)进行数据库操作

步骤 1: 添加依赖 在 pom.xml 文件中添加相关依赖&#xff1a; <dependencies><!-- Spring Boot Starter Web --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><…...

简单聊聊PLT和GOT

在 Linux 的动态链接中&#xff0c;PLT&#xff08;Procedure Linkage Table&#xff09; 和 GOT&#xff08;Global Offset Table&#xff09; 是动态链接机制中的两个关键组件&#xff0c;它们一起支持程序动态加载共享库以及在运行时解析符号地址。下面是它们的作用和原理&a…...

FaRM译文

No compromises: distributed transactions with consistency, availability, and performance Aleksandar Dragojevic, Dushyanth Narayanan, Edmund B. Nightingale, Matthew Renzelmann, Alex Shamis, Anirudh Badam, Miguel Castro Microsoft Research 目录 摘要 1. 引…...

用vue框架写一个时钟的页面

你可以使用Vue框架来创建一个简单的时钟页面。首先&#xff0c;你需要在HTML文件中引入Vue框架的CDN&#xff1a; <script src"https://cdn.jsdelivr.net/npm/vue"></script>然后&#xff0c;创建一个包含时钟功能的Vue实例&#xff1a; <div id&qu…...

HTML表单-第二部分

HTML表单 表单元素是允许用户在表单中输入内容&#xff0c;比如&#xff1a;文本域&#xff0c;下拉列表&#xff0c;单选框&#xff0c;复选框等等‘ 使用<from>标签创建 例如 <from> . input . </from> HTML表单-输入元素 <input>标签创建&#xff…...

PyQt5:一个逗号引发的闪退血案

【日常小计】 在开发PyQt5程序时&#xff0c;调用了一个写入excel表格的后端方法&#xff0c;但是每次打开页面点击对应的动作&#xff0c;窗口就会闪退&#xff0c;而且Python后台也没有提示出任何的异常堆栈&#xff0c;后来经过在后端一点一点的单点测试&#xff0c;终于发…...

AI智能体Prompt预设词指令大全+GPTs应用使用

AI智能体使用指南 直接复制在AI工具助手中使用&#xff08;提问前&#xff09; 可前往SparkAi系统用户官网进行直接使用 SparkAI系统介绍文档&#xff1a;Docs 常见AI智能体GPTs应用大全在线使用 自定义添加制作AI智能体进行使用&#xff1a; 文章润色器 你是一位具有敏锐洞察…...

SSM整合原理实战案例《任务列表案例》

一、前端程序搭建和运行: 1.整合案例介绍和接口分析: (1).案例功能预览: (2).接口分析: 学习计划分页查询 /* 需求说明查询全部数据页数据 请求urischedule/{pageSize}/{currentPage} 请求方式 get 响应的json{"code":200,"flag":true,"data&…...

在风能市场持续增长的情况下,全球【环氧活性稀释剂】的需求呈现明显上涨的趋势

摘要 据 HengCe 最新调研&#xff0c;2023年中国环氧活性稀释剂市场销售收入达到了 万元&#xff0c;预计2030年可以达到 万元&#xff0c;2024-2030期间年复合增长率(CAGR)为 %。本研究项目旨在梳理环氧活性稀释剂领域产品系列&#xff0c;洞悉行业特点、市场存量空间及增量空…...

CSS一些小点 —— 12.7

1. box-sizing: border-box box-sizing 属性&#xff0c;默认值为 content-box box-sizing: border-box 使padding和border的值不会再影响元素的宽高&#xff1b;padding和border的值算在指定宽高的内部&#xff08;但是外边距依然算做外部&#xff09; 2. overflow: hidden …...

[NeurlPS 2022] STaR 开源代码实现解读

STaR 方法代码开源&#xff0c;这里给出一个中文代码解读地址&#xff1a;repo入口点&#xff1a;iteration_train.py&#xff1b;关键代码&#xff1a;device_train.py, device_inference.py, and create_finetune_tfrecords.py&#xff1b;基于 JAX、RAY&#xff0c;在 Googl…...