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

二叉树的层次遍历经典问题-算法通关村

二叉树的层次遍历经典问题-算法通关村


1 层次遍历简介

  • 广度优先在面试里出现的频率非常高,整体属于简单题。广度优先又叫层次遍历,基本过程如下:

  • 层次遍历就是从根节点开始,先访问根节点下面一层全部元素,再访问之后的层次,类似金字塔一样一层层访问。我们可以看到这里就是从左到右一层一层的去遍历二叉树,先访问3,之后访问3的左右子孩子9和20,之后分别访问9和20的左右子孩子[8,13]和[15,17],最后得到结果
    [3,9,20,8,13,15,17]。
    这里的问题是怎么将遍历过的元素的子孩子保存一下呢,例如访问9时其左右子孩子8和13应该先存一下,直到处理20之后才会处理。使用队列来存储能完美解决上述问题,例如上面的图中:

  • 1. 首先3入队 2.然后3出队,之后将3的左右孩子9和20,保存到队列中。 3.之后9出队,将9的左右孩子8和13入队。 4.之后20出队,将20的左右孩子15和7入队。 5.之后 8,13,15,7分别出队,此时都是叶子结点,只出队就行。
  • 这就是LeetCode里经典的层次遍历题!102.二叉树的层序遍历,107.二叉树的层次遍历II,199.二叉树的右视图,637.二叉树的层平均值,429.N叉树的前序遍历,515.在每个树行中找最大值,116.填充每个节点的下一个右侧节点指针,117.填充每个节点的下一个右侧节点指针II,103 锯齿层序遍历
    除此之外,在深度优先的题目里,有些仍然会考虑层次遍历的实现方法。


2 基本的层序遍历与变换

  • 仅仅遍历并输出全部元素

  •   public List<Integer> simpleLevelOrder(TreeNode root){if(root == null){return new ArrayList<Integer>();}List<Integer> list = new ArrayList<>();Deque<TreeNode> TreeQueue = new ArrayDeque<>();//将根节点放入队列中,然后不断遍历TreeQueue.offer(root);while(!TreeQueue.isEmpty()){//一层一层的遍历for(int i = 0; i< TreeQueue.size(); i++){TreeNode temp = TreeQueue.poll();list.add(temp.val);if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}}return list;}
    

2.1二叉树的层序遍历

  • LeetCode102:给你一个二叉树,请你返回其按层次遍历得到的节点值。(即逐层地,从左到右访问所有节点)。

  • 如何判断某一层访问完了呢?简单,用一个变量size标记一下就行了,size表示某一层的元素个数,只要出队,就将size减1,减到O就说明该层元素访问完了。当size变成0之后,这时队列中剩余元素的个数恰好就是下一层元素的个数,因此重新将size标记为下一层的元素个数就可以继续处理新的一行了。最后,把每层遍历到的节点都放到一个结果集中,将其返回就行了。

  •   public List<List<Integer>> levelOrder(TreeNode root){if(root == null){return new ArrayList<List<Integer>>();}List<List<Integer>> res = new ArrayList<>();Deque<TreeNode> TreeQueue = new ArrayDeque<>();//将根节点放入队列中,然后不断遍历TreeQueue.offer(root);while(!TreeQueue.isEmpty()){//存储每一层的元素List<Integer> list = new ArrayList<>();for(int i = 0; i< TreeQueue.size(); i++){TreeNode temp = TreeQueue.poll();list.add(temp.val);if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}res.add(list);}return res;}
    

2.2 层序遍历-自底向上

  • LeetCode 107.给定一个二叉树,返回其节点值自底向上的层序遍历。(即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历)。例如给定的二叉树为:

  • [ [15, 7], [9, 20], [3] ]
  • 如果要求从上到下输出每一层的节点值,做法是很直观的,在遍历完一层节点之后,将存储该层节点值的列表添加到结果列表的尾部。这道题要求从下到上输出每一层的节点值,只要对上述操作稍作修改即可,在遍历完一层节点之后,将存储该层节点值的列表添加到结果列表的头部

  •   public List<List<Integer>> levelOrderBottom(TreeNode root){if(root == null){return new ArrayList<List<Integer>>();}List<List<Integer>> res = new ArrayList<>();Deque<TreeNode> TreeQueue = new ArrayDeque<>();//将根节点放入队列中,然后不断遍历TreeQueue.offer(root);while(!TreeQueue.isEmpty()){//存储每一层的元素List<Integer> levelList = new ArrayList<>();for(int i = 0; i< TreeQueue.size(); i++){TreeNode temp = TreeQueue.poll();levelList.add(temp.val);if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}//当前层的节点值列表 levelList 添加到结果列表 res 的开头,// 实现了层次遍历结果的逆序存储。res.add(0, levelList);}return res;}
    

2.3二叉树的锯齿形层序遍历

  • LeetCode103 :给定一个二叉树,返回其节点值的锯齿形层序遍历。(即先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行)。
    例如:给定二叉树 [3,9,20,null,null,15,7]

  • [ [3], [20, 9] [15, 7] ]
  • 这个题也是102的变种,只是最后输出的要求有所变化,要求我们按层数的奇偶来决定每一层的输出顺序。如果当前层数是偶数,从左至右输出当前层的节点值,否则,从右至左输出当前层的节点值。

  • 我们依然可以沿用第102题的思想,为了满足题目要求的返回值为**「先从左往右,再从右往左」交替输出的锯齿形,可以利用「双端队列」**的数据结构来维护当前层节点值输出的顺序。双端队列是一个可以在队列任意一端插入元素的队列。在广度优先搜索遍历当前层节点拓展下一层节点的时候我们仍然从左往右按顺序拓展,但是对当前层节点的存储我们维护一个变量isOrderLeft
    记录是从左至右还是从右至左的:
    • 如果从左至右,我们每次将被遍历到的元素插入至双端队列的末尾。
    •从右至左,我们每次将被遍历到的元素插入至双端队列的头部。

  •   public List<List<Integer>> zigzagLevelOrder(TreeNode root){if(root == null){return new ArrayList<List<Integer>>();}List<List<Integer>> res = new ArrayList<>();Deque<TreeNode> TreeQueue = new ArrayDeque<>();//将根节点放入队列中,然后不断遍历TreeQueue.offer(root);boolean isOrderLeft = true;while(!TreeQueue.isEmpty()){//存储每一层的元素Deque<Integer> levelQueue = new ArrayDeque<>();for(int i = 0; i< TreeQueue.size(); i++){TreeNode temp = TreeQueue.poll();if(isOrderLeft){levelQueue.addLast(temp.val);}else{levelQueue.addFirst(temp.val);}if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}res.add(new ArrayList<Integer>(levelQueue));isOrderLeft = !isOrderLeft;}return res;}
    

2.4 N 叉树的层序遍历

  • LeetCode429 给定一个 N 叉树,返回其节点值的层序遍历。(即从左到右,逐层遍历)。树的序列化输入是用层序遍历,每组子节点都由 null值分隔(参见示例)。

  • 输入:root = [ 1, null, 3, 2, 4, null, 5, 6](表述树的元素是这个序列) 输出:[ [1] ,[3,2,4],[5,61] ]
  • N 叉树的定义如下,就是一个值,加一个列表。

  •   public class Node {public int val;public List<Node> children;}
    
  •   public List<List<Integer>> nLevelOrder(Node root){if(root == null){return new ArrayList<List<Integer>>();}List<List<Integer>> res = new ArrayList<>();Deque<Node> NTreeQueue = new ArrayDeque<>();NTreeQueue.offer(root);while(!NTreeQueue.isEmpty()){List<Integer> levelList = new ArrayList<>();while(!NTreeQueue.isEmpty()){Node cur = NTreeQueue.pollFirst();levelList.add(cur.val);for (Node child : cur.children) {if(child != null){NTreeQueue.add(child);}}}res.add(levelList);}return res;}
    

3 几个处理每层元素的项目

  • LeetCode三道题目:515.在每个树行中找最大值(最小),637.二叉树的层平均值,199.二叉树的右视图。

3.1 在每个树行中找最大值

  • LeetCode515: 给定一棵二叉树的根节点 root,请找出该二叉树中每一层的最大值。

  • 在得到一层的元素之后,用一个变量记录其最大值。

  •   public List<Integer> largestValues(TreeNode root){if(root == null){return new ArrayList<Integer>();}List<Integer> res = new ArrayList<>();Deque<TreeNode> TreeQueue = new ArrayDeque<>();//将根节点放入队列中,然后不断遍历TreeQueue.offer(root);while(!TreeQueue.isEmpty()){//存储每一层的元素int levelMaxNum = Integer.MIN_VALUE;for(int i = 0; i< TreeQueue.size(); i++){TreeNode temp = TreeQueue.poll();levelMaxNum = Math.max(levelMaxNum, temp.val);if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}res.add(levelMaxNum);}return res;}
    

3.2 在每个树行中找平均值

  • LeetCode637:要求给定一个非空二叉树,返回一个由每层节点平均值组成的数组。

  • 将每层的元素都先保存下来,最后求平均值。

  •   public List<Double> averageOfLevels(TreeNode root){if(root == null){return new ArrayList<Double>();}List<Double> res = new ArrayList<>();Deque<TreeNode> TreeQueue = new ArrayDeque<>();//将根节点放入队列中,然后不断遍历TreeQueue.offer(root);while(!TreeQueue.isEmpty()){double levelAverage = 0;int len = TreeQueue.size();for(int i = 0; i< len; i++){TreeNode temp = TreeQueue.poll();levelAverage += temp.val;if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}res.add(levelAverage/len);}return res;}
    

3.3 二叉树的右视图

  • LeetCode 199:给定一个二叉树的根节点 root,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。例如:

  • 利用BFS进行层次遍历,记录下每层的最后一个元素。

  •   public List<Integer> rightSideView(TreeNode root){if(root == null){return new ArrayList<Integer>();}List<Integer> res = new ArrayList<>();Deque<TreeNode> TreeQueue = new ArrayDeque<>();//将根节点放入队列中,然后不断遍历TreeQueue.offer(root);while(!TreeQueue.isEmpty()){int len = TreeQueue.size();int rightValue = 0;for(int i = 0; i< len; i++){TreeNode temp = TreeQueue.poll();if(i == len-1){rightValue = temp.val;}if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}res.add(rightValue);}return res;}
    

3.4 最底层最左边

  • LeetCode513: 给定一个二叉树的 根节点root,请找出该二叉树的 最底层 最左边 节点的值。

  • 假设二叉树中至少有一个节点。

  • 示例1:
    输入:root = [2, 1, 3] 输出:1
    示例2:
    输入:[1, 2, 3, 4, null, 5, 6, null, null, 7] 输出:7
  • 这里有两个问题:该怎么知道什么时候到了最底层呢?假如最底层有两个,该怎么知道哪个是最左的呢?我们继续观察层次遍历的执行过程:

  • 我们可以发现,正常执行层次遍历,不管最底层有几个元素,最后一个输出的一定是是最底层最右的元素7,那这里我们就想了,能否将该处理与上一次题的翻转结合一下,每一层都是先反转再放入队列,就可以让最后一个输出的是最左的呢?是的,这就是解决本题的关键。

  •   public int findBottomLeftValue(TreeNode root){if(root.right == null && root.left == null){return root.val;}Deque<TreeNode> TreeQueue = new ArrayDeque<>();TreeQueue.offer(root);TreeNode temp = new TreeNode();while(!TreeQueue.isEmpty()){int len = TreeQueue.size();for(int i = 0; i< len; i++){temp = TreeQueue.poll();//右节点先入队if(temp.right != null){TreeQueue.offer(temp.right);}//左节点后入队if(temp.left != null){TreeQueue.offer(temp.left);}}}return temp.val;}
    
  • 下面这个没有上面简洁,而且相比之下增加了一些额外的开销,仅供参考:

  •   public int findBottomLeftValue2(TreeNode root){if(root.right == null && root.left == null){return root.val;}Deque<TreeNode> TreeQueue = new ArrayDeque<>();TreeQueue.offer(root);int res = 0;while(!TreeQueue.isEmpty()){int len = TreeQueue.size();Deque<TreeNode> queue = new ArrayDeque<>();for(int i = 0; i< len; i++){TreeNode temp = TreeQueue.poll();queue.addFirst(temp);if(temp.left != null){TreeQueue.offer(temp.left);}if(temp.right != null){TreeQueue.offer(temp.right);}}res = queue.removeLast().val;}return res;}
    

相关文章:

二叉树的层次遍历经典问题-算法通关村

二叉树的层次遍历经典问题-算法通关村 1 层次遍历简介 广度优先在面试里出现的频率非常高&#xff0c;整体属于简单题。广度优先又叫层次遍历&#xff0c;基本过程如下&#xff1a; 层次遍历就是从根节点开始&#xff0c;先访问根节点下面一层全部元素&#xff0c;再访问之后…...

SQLiteC/C++接口详细介绍sqlite3_stmt类(十二)

返回&#xff1a;SQLite—系列文章目录 上一篇&#xff1a;SQLiteC/C接口详细介绍sqlite3_stmt类&#xff08;十一&#xff09; 下一篇&#xff1a; SQLiteC/C接口详细介绍sqlite3_stmt类&#xff08;十三&#xff09; 48、sqlite3_stmt_isexplain sqlite3_stmt_is…...

大模型时代如何做安全?

现在应该没人怀疑AI时代的到来了吧&#xff0c;在HUB上每天100的新的预训练模型产生&#xff0c;不夸张的说的&#xff0c;现在稍微有点计算机基础的人都可以训练自己的模型了。 说远了&#xff0c;还是说说那些不争气的安全厂商吧。为啥只说安全厂商&#xff1f;因为国内还是…...

新型储能是什么,储能系统解决方案现状及趋势详细说明

新型储能是指新兴的能够存储电能并在需要时释放的储能技术。其中主要包括光伏储能和商业储能。 光伏储能是指通过光伏电池将太阳能转化为电能&#xff0c;并将其存储起来以供后续使用。光伏储能系统一般由太阳能电池板、储能装置和逆变器组成。光伏储能可以将白天产生的电能存…...

掌握Go语言:Go语言中的字典魔法,高效数据检索与应用实例解析(18)

在Go语言中&#xff0c;字典通常指的是map类型&#xff0c;它是一种用于存储键值对的数据结构。字典在Go中非常常见&#xff0c;是一种高效的数据结构&#xff0c;用于快速查找和检索数据。 字典的详细使用方法 创建字典 可以使用make函数来创建字典&#xff0c;并指定键值对…...

Flutter-仿携程首页类型切换

效果 唠叨 闲来无事&#xff0c;不小心下载了携程app&#xff0c;还幻想可以去旅游一番&#xff0c;奈何自己运气不好&#xff0c;自从高考时第一次吹空调导致自己拉肚子考试&#xff0c;物理&#xff0c;数学考了一半就交卷&#xff0c;英语2B铅笔除了问题&#xff0c;导致原…...

C语言 自定义类型:结构体

目录 前言 一、结构体类型 1.1 结构体的声明 1.2 结构体变量的创建和初始化 1.3 结构体的特殊声明 1.4 结构体的自引用 二、结构体的对齐 2.1 对齐规则 2.2 内存对齐的原因 2.3 修改默认对齐数 2.4 结构体传参 三、结构体实现位段 3.1 位段的内存分配 3.2 段的跨平…...

计算机网络拓扑结构

目录 <网络拓扑结构概念> <典型的拓扑结构介绍> 第一种&#xff0c;总线型网络拓扑结构 第二种&#xff0c;星型网络拓扑结构 第三种&#xff0c;树型网络拓扑结构 第四种&#xff0c;环型网络拓扑结构 第五种&#xff0c;网状型网络拓扑结构 第六种&#…...

FPGA通过I2C控制AT24C64

文章目录 前言一、代码设计框图二、IIC_drive模块设计2.1、模块接口&#xff1a;2.2、代码功能描述&#xff1a;2.3、IIC协议实现过程&#xff1a; 三、EEPROM_ctrl模块设计3.1、模块接口&#xff1a;3.2、代码功能描述 四、EEPROM_drive模块五、iic_top模块 前言 继上一篇FPG…...

134. 加油站(力扣LeetCode)

文章目录 134. 加油站题目描述暴力枚举&#xff08;超时&#xff09;代码一代码二&#xff08;优化&#xff09; 贪心算法方法一方法二 134. 加油站 题目描述 在一条环路上有 n 个加油站&#xff0c;其中第 i 个加油站有汽油 gas[i] 升。 你有一辆油箱容量无限的的汽车&…...

XSKY 智能存储,助力“数据要素 X”先进制造

3 月 21-22 日&#xff0c;主题为“突破 智行”的 IMC2024 第七届中国智造数字科技峰会在重庆召开。作为在先进制造领域拥有领先存储解决方案以及众多应用实践的企业&#xff0c;星辰天合受邀参加了此次峰会并荣获大会颁发的“最佳存储解决方案奖”。同时&#xff0c;星辰天合先…...

数据挖掘与分析学习笔记

一、Numpy NumPy&#xff08;Numerical Python&#xff09;是一种开源的Python库&#xff0c;专注于数值计算和处理多维数组。它是Python数据科学和机器学习生态系统的基础工具包之一&#xff0c;因为它高效地实现了向量化计算&#xff0c;并提供了对大型多维数组和矩阵的支持…...

linux docker镜像初始化

linux docker镜像初始化 简介 有的镜像内部使用的linux系统特别精简&#xff0c;许多常用命令无法安装&#xff0c;导致排查问题较为困难。 可以使用cat /etc/os-release查看容器使用的linux版本&#xff0c;再进行一些常用操作的初始化。 Debian # 设置镜像源 RUN rm -f /…...

专业140+总分410+南京大学851信号与系统考研经验南大电子信息与通信集成,电通,真题,大纲,参考书。

今年分数出来还是有点小激动&#xff0c;专业851信号与系统140&#xff08;感谢Jenny老师辅导和全程悉心指导&#xff0c;答疑&#xff09;&#xff0c;总分410&#xff0c;梦想的南大离自己越来越近&#xff0c;马上即将复试&#xff0c;心中慌的一p&#xff0c;闲暇之余&…...

. ./ bash dash source 这五种执行shell脚本方式 区别

实际上,., ./, bash, dash, source 是五种不同的方式来执行 shell 脚本,它们之间有一些区别。 .(点号)或 source 命令:这两个命令是等价的,它们都是 Bash shell 内置的命令。它们用于在当前 shell 环境中执行脚本。当使用 . script.sh 或 source script.sh 命令来执行脚本…...

【React 】React 性能优化的手段有哪些?

1. 是什么 React凭借virtual DOM和diff算法拥有高效的性能&#xff0c;但是某些情况下&#xff0c;性能明显可以进一步提高 在前面文章中&#xff0c;我们了解到类组件通过调用setState方法&#xff0c;就会导致render ,父组件一旦发生render渲染&#xff0c;子组件一定也会执…...

3.22网络编程小项目

基于UDP的网络聊天室 项目需求&#xff1a; 如果有用户登录&#xff0c;其他用户可以收到这个人的登录信息如果有人发送信息&#xff0c;其他用户可以收到这个人的群聊信息如果有人下线&#xff0c;其他用户可以收到这个人的下线信息服务器可以发送系统信息 服务器 #includ…...

Git原理及使用

1、Git初识 Git是一种版本控制器: 对于同一份文件,做多次改动,Git会记录每一次改动前后的文件。 通俗的讲就是⼀个可以记录⼯程的每⼀次改动和版本迭代的⼀个管理系统,同时也⽅便多⼈协同作业。 注意: Git其实只能跟踪⽂本⽂件的改动,⽐如TXT⽂件,⽹⻚,所有的程序代码…...

Milvus 向量数据库介绍及使用

一、Milvus 介绍及安装 Milvus 于 2019 年创建&#xff0c;其目标只有一个&#xff1a;存储、索引和管理由深度神经网络和其他机器学习 (ML) 模型生成的大量嵌入向量。它具备高可用、高性能、易拓展的特点&#xff0c;用于海量向量数据的实时召回。 作为专门为处理输入向量查…...

STP环路避免实验(华为)

思科设备参考&#xff1a;STP环路避免实验&#xff08;思科&#xff09; 一&#xff0c;技术简介 Spanning Tree Protocol&#xff08;STP&#xff09;&#xff0c;即生成树协议&#xff0c;是一种数据链路层协议。主要作用是防止二层环路&#xff0c;并自适应网络变化和故障…...

【杂谈】-递归进化:人工智能的自我改进与监管挑战

递归进化&#xff1a;人工智能的自我改进与监管挑战 文章目录 递归进化&#xff1a;人工智能的自我改进与监管挑战1、自我改进型人工智能的崛起2、人工智能如何挑战人类监管&#xff1f;3、确保人工智能受控的策略4、人类在人工智能发展中的角色5、平衡自主性与控制力6、总结与…...

相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解

【关注我&#xff0c;后续持续新增专题博文&#xff0c;谢谢&#xff01;&#xff01;&#xff01;】 上一篇我们讲了&#xff1a; 这一篇我们开始讲&#xff1a; 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下&#xff1a; 一、场景操作步骤 操作步…...

【C语言练习】080. 使用C语言实现简单的数据库操作

080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...

零基础在实践中学习网络安全-皮卡丘靶场(第九期-Unsafe Fileupload模块)(yakit方式)

本期内容并不是很难&#xff0c;相信大家会学的很愉快&#xff0c;当然对于有后端基础的朋友来说&#xff0c;本期内容更加容易了解&#xff0c;当然没有基础的也别担心&#xff0c;本期内容会详细解释有关内容 本期用到的软件&#xff1a;yakit&#xff08;因为经过之前好多期…...

NXP S32K146 T-Box 携手 SD NAND(贴片式TF卡):驱动汽车智能革新的黄金组合

在汽车智能化的汹涌浪潮中&#xff0c;车辆不再仅仅是传统的交通工具&#xff0c;而是逐步演变为高度智能的移动终端。这一转变的核心支撑&#xff0c;来自于车内关键技术的深度融合与协同创新。车载远程信息处理盒&#xff08;T-Box&#xff09;方案&#xff1a;NXP S32K146 与…...

【前端异常】JavaScript错误处理:分析 Uncaught (in promise) error

在前端开发中&#xff0c;JavaScript 异常是不可避免的。随着现代前端应用越来越多地使用异步操作&#xff08;如 Promise、async/await 等&#xff09;&#xff0c;开发者常常会遇到 Uncaught (in promise) error 错误。这个错误是由于未正确处理 Promise 的拒绝&#xff08;r…...

Python 训练营打卡 Day 47

注意力热力图可视化 在day 46代码的基础上&#xff0c;对比不同卷积层热力图可视化的结果 import torch import torch.nn as nn import torch.optim as optim from torchvision import datasets, transforms from torch.utils.data import DataLoader import matplotlib.pypl…...

FFmpeg avformat_open_input函数分析

函数内部的总体流程如下&#xff1a; avformat_open_input 精简后的代码如下&#xff1a; int avformat_open_input(AVFormatContext **ps, const char *filename,ff_const59 AVInputFormat *fmt, AVDictionary **options) {AVFormatContext *s *ps;int i, ret 0;AVDictio…...

DiscuzX3.5发帖json api

参考文章&#xff1a;PHP实现独立Discuz站外发帖(直连操作数据库)_discuz 发帖api-CSDN博客 简单改造了一下&#xff0c;适配我自己的需求 有一个站点存在多个采集站&#xff0c;我想通过主站拿标题&#xff0c;采集站拿内容 使用到的sql如下 CREATE TABLE pre_forum_post_…...

【安全篇】金刚不坏之身:整合 Spring Security + JWT 实现无状态认证与授权

摘要 本文是《Spring Boot 实战派》系列的第四篇。我们将直面所有 Web 应用都无法回避的核心问题&#xff1a;安全。文章将详细阐述认证&#xff08;Authentication) 与授权&#xff08;Authorization的核心概念&#xff0c;对比传统 Session-Cookie 与现代 JWT&#xff08;JS…...