深度优先搜索(DFS)与回溯法:从全排列到子集问题的决策树与剪枝优化

在这里插入图片描述

文章目录

  • 前言
    • 🎄一、全排列
      • ✨核心思路
      • ✨实现步骤
      • ✨代码
      • ✨时间和空间复杂度
        • 🎁1. 时间复杂度
        • 🎁2. 空间复杂度
    • 🎄二、子集
      • ✨解法一:逐位置决策法
        • 🎁步骤分析
        • 🎁运行示例
        • 🎁代码
      • ✨解法二:逐元素扩展法
        • 🎁步骤分析
        • 🎁运行示例
        • 🎁代码
      • ✨时间复杂度分析
        • 🎁解法一
        • 🎁解法二
      • ✨空间复杂度分析
        • 🎁解法一
        • 🎁解法二
    • 🎄三、找出所有子集的异或总和再求和
      • ✨解法简介
      • ✨解法步骤
        • 🎁Step 1: 初始化变量
        • 🎁Step 2: 回溯递归函数设计
        • 🎁Step 3: 主函数调用
      • ✨代码
      • ✨时间与空间复杂度
        • 🎁时间复杂度
        • 🎁空间复杂度
    • 🎄四、全排列II
      • ✨解法简介
      • ✨步骤详解
        • 🎁Step 1: 初始化
        • 🎁Step 2: 回溯递归函数设计
        • 🎁Step 3: 主函数调用
      • ✨代码
      • ✨时间与空间复杂度
        • 🎁时间复杂度
        • 🎁空间复杂度
  • 结语


前言

深度优先搜索(DFS)和回溯法是解决复杂问题中不可或缺的算法工具,尤其在组合问题(如全排列、子集等)中,发挥着至关重要的作用。通过递归的方式,DFS 能够遍历问题的解空间,而回溯法则通过撤销不合法的选择,避免重复计算,提高效率。在解题过程中,剪枝是优化回溯法的重要手段,它通过提前排除无效路径,进一步减少了运算的复杂度。本文将深入探讨如何使用 DFS、回溯法及剪枝技术,构建解决全排列和子集问题的决策树,并优化算法的执行效率。


🎄一、全排列

题目链接:https://leetcode.cn/problems/permutations/description/

✨核心思路

这段代码实现了生成一个数组的所有排列(Permutation),并返回所有可能的排列列表。采用了**深度优先搜索(DFS)**的方式,配合一个 check 数组来记录哪些元素已经使用过,从而避免重复使用。

✨实现步骤

  1. 初始化变量
    • vector<vector<int>> ret:存储最终结果,包含所有排列。
    • vector<int> path:存储当前正在构造的排列。
    • bool check[7]:标记数组,记录某个位置是否已经被使用,避免重复选择。默认全为 false
  2. 主函数 permute
    • 调用 dfs(nums) 开始进行递归遍历,寻找所有排列。
    • 返回最终结果 ret
  3. 辅助函数 dfs
    • 递归终止条件:当 path 的长度等于 nums.size(),说明已经构造出一个完整的排列,将其加入结果集 ret
    • 循环遍历:
      • 遍历数组的每个元素,判断其是否已经被使用(check[i] == false)。
      • 如果未使用,将其加入当前路径 path,同时更新 check[i] = true
      • 递归调用 dfs(nums),继续构造排列。
      • 回溯:
        • 移除当前路径中的最后一个元素 path.pop_back()
        • 重置 check[i] = false,允许后续尝试其他路径。

如图分析:

在这里插入图片描述

✨代码

class Solution {
public:
    vector<vector<int>> ret;
    vector<int> path;
    bool check[7];
    vector<vector<int>> permute(vector<int>& nums) {
        dfs(nums);
        return ret;
    }
    void dfs(vector<int>& nums) {
        if (path.size() == nums.size()) {
            ret.push_back(path);
            return;
        }
        for (int i = 0; i < nums.size(); i++) {
            if (check[i] == false) {
                path.push_back(nums[i]);
                check[i] = true;
                dfs(nums);
                //  -> ֳָ
                path.pop_back();
                check[i] = false;
            }
        }
    }
};

✨时间和空间复杂度

🎁1. 时间复杂度
  • 每个排列需要遍历 nums 的所有元素,同时需要递归构造排列。
  • 全排列的总数为 n ! n! n! ,其中 n 为 nums 的长度。
  • 每次构造排列需要 O ( n ) O(n) O(n) 的时间复杂度,因此总时间复杂度为: O ( n ⋅ n ! ) O(n⋅n!) O(nn!)
🎁2. 空间复杂度
  • 递归深度为 O ( n ) O(n) O(n),对应递归栈的最大深度。
  • 存储结果的 ret 大小为 O ( n ⋅ n ! ) O(n⋅n!) O(nn!)(每个排列长度为 n,共有 n! 个排列)。
  • 额外空间为 pathcheck,均为 $O(n) $。
  • 总空间复杂度为: O ( n ⋅ n ! ) O(n⋅n!) O(nn!)

🎄二、子集

题目链接:https://leetcode.cn/problems/subsets/description/

✨解法一:逐位置决策法

核心思想:对于每个位置,都需要明确“选”或者“不选”两种可能,然后继续对后续位置进行递归。

🎁步骤分析
  1. 输入数组nums = [1, 2, 3]
  2. 决策树构造:
    • 对于每个元素,存在两个分支:选它不选它
    • 递归过程中,我们先“选”再“回溯”,然后“跳过当前元素”。
  3. 递归终止条件:
    • 当递归位置 pos 达到数组末尾(pos == nums.size())时,将当前路径 path 加入结果集 ret
🎁运行示例
  • 初始调用:dfs(nums, 0)
    • 选1:path = [1],递归进入 dfs(nums, 1)
      • 选2:path = [1, 2],递归进入 dfs(nums, 2)
        • 选3path = [1, 2, 3],递归终止,保存结果 [[1, 2, 3]]
        • 回溯到选2状态path = [1, 2]
        • 不选3path = [1, 2],递归终止,保存结果 [[1, 2, 3], [1, 2]]
      • 回溯到选1状态path = [1]
      • 不选2:path = [1],递归进入 dfs(nums, 2)
        • 选3path = [1, 3],递归终止,保存结果 [[1, 2, 3], [1, 2], [1, 3]]
        • 不选3path = [1],递归终止,保存结果 [[1, 2, 3], [1, 2], [1, 3], [1]]
    • 回溯到初始状态path = []
    • 不选1:path = [],递归进入 dfs(nums, 1)
      • 选2path = [2],递归类似上面。
      • 不选2path = [],递归继续处理后续。

最终得到所有子集。

如图分析:

在这里插入图片描述

🎁代码
class Solution {
public:
    vector<vector<int>> ret;
    vector<int> path;
    vector<vector<int>> subsets(vector<int>& nums) {
        dfs(nums, 0);
        return ret;
    }
    void dfs(vector<int>& nums, int pos) {
        if (pos == nums.size()) {
            ret.push_back(path);
            return;
        }
        // 选
        path.push_back(nums[pos]);
        dfs(nums, pos + 1);
        // 回溯 -> 恢复现场
        path.pop_back();

        // 不选
        dfs(nums, pos + 1);
    }
};

✨解法二:逐元素扩展法

核心思想:直接将当前路径加入结果集,然后从当前位置逐个向后选取元素递归。

🎁步骤分析
  1. 输入数组nums = [1, 2, 3]
  2. 递归机制:
    • 每次递归时,直接将当前路径加入结果集。
    • 从当前位置 pos 开始,依次向后选择元素,尝试递归。
🎁运行示例
  • 初始调用:dfs(nums, 0)
    • 当前路径为空,加入结果集:ret = [[]]
    • pos = 0 开始:
      • 选1:path = [1],加入结果集:ret = [[], [1]],递归进入 dfs(nums, 1)
        • 选2:path = [1, 2],加入结果集:ret = [[], [1], [1, 2]],递归进入 dfs(nums, 2)
          • 选3path = [1, 2, 3],加入结果集:ret = [[], [1], [1, 2], [1, 2, 3]]
          • 回溯到选2状态path = [1, 2]
        • 回溯到选1状态path = [1]
      • 回溯到初始状态path = []
      • 选2:递归同上。
      • 选3:递归继续。

如图分析:

在这里插入图片描述

🎁代码
class Solution {
public:
    vector<vector<int>> ret;
    vector<int> path;
    vector<vector<int>> subsets(vector<int>& nums) {
        dfs(nums, 0);
        return ret;
    }
    void dfs(vector<int>& nums, int pos) {
        ret.push_back(path);
        for (int i = pos; i < nums.size(); i++) {
            path.push_back(nums[i]);
            dfs(nums, i + 1);
            // 回溯 -> 恢复现场
            path.pop_back();
        }
    }
};

✨时间复杂度分析

🎁解法一
  1. 决策树分析
    • 每个元素有两种选择(选或不选),对于长度为 n 的数组,共有 2^n 种子集。
    • 递归的每一步都对应一个子集的生成,因此遍历整个决策树需要 O(2^n) 的时间。
  2. 额外操作
    • 每次递归终止时,将当前路径 path 加入结果,需要 O(n) 的时间(复制 path 的内容)。

总时间复杂度
O ( 2 n ⋅ n ) O(2^n⋅n) O(2nn)

🎁解法二
  1. 决策树分析
    • 每个元素被递归遍历一次,所有可能的子集共 2^n 个。
    • 递归过程中,每次递归都会将当前路径 path 加入结果集 ret
  2. 额外操作
    • 每次将路径加入结果集也需要 O(n) 的时间(复制 path)。

总时间复杂度
O(2n⋅n)


✨空间复杂度分析

🎁解法一
  1. 递归深度
    • 递归调用的最大深度为数组长度 n ,因此递归栈的空间复杂度为 O ( n ) O(n) O(n)
  2. 路径存储
    • path 需要存储当前路径,大小为 O ( n ) O(n) O(n)
  3. 结果集
    • 最终结果集 ret 包含 2 n 2^n 2n 个子集,每个子集的平均长度为 O ( n / 2 ) O(n/2) O(n/2),因此结果集的存储空间为: O ( 2 n ⋅ n ) O(2n⋅n) O(2nn)

总空间复杂度
O ( 2 n ⋅ n ) O(2^n \cdot n) O(2nn)
(递归栈和路径空间较小,可忽略不计)

🎁解法二
  1. 递归深度
    • 递归调用的最大深度为数组长度 n,因此递归栈的空间复杂度为 O ( n ) O(n) O(n)
  2. 路径存储
    • path 需要存储当前路径,大小为 O ( n ) O(n) O(n)
  3. 结果集
    • 同解法一,结果集需要 $O(2^n \cdot n) $的空间。

总空间复杂度
O ( 2 n ⋅ n ) O(2^n \cdot n) O(2nn)
(递归栈和路径空间较小,可忽略不计)

🎄三、找出所有子集的异或总和再求和

题目链接:https://leetcode.cn/problems/sum-of-all-subset-xor-totals/description/

✨解法简介

题目要求求出数组中所有子集的异或和的总和。解法采用 回溯法 (Backtracking),枚举所有子集,同时计算每个子集的异或和,并将其累加到结果中。


✨解法步骤

🎁Step 1: 初始化变量
  • ret:最终结果,用于存储所有子集的异或和的累加值。
  • path:当前路径变量,表示当前子集的异或值。
  • nums:输入数组。

🎁Step 2: 回溯递归函数设计

递归函数 dfs(nums, pos)

  1. 功能
    • 遍历以 pos 为起点的所有可能子集。
    • 累加每个子集的异或值到 ret
  2. 核心逻辑
    • 累加当前路径的异或值到 ret: 每次递归进入时,将当前路径的异或值(path)加入到结果中。
    • 枚举后续元素: 从当前 pos 开始,逐一尝试将后续元素加入路径 path,并递归处理。
    • 回溯: 在递归返回后,撤销当前选择(通过异或运算恢复 path 的值)。
  3. 递归终止条件
    • 无需显式终止条件,因为递归自然会遍历所有子集,当 pos 超出数组范围时,循环自动结束。

🎁Step 3: 主函数调用

在主函数 subsetXORSum 中:

  1. 初始化结果变量 ret = 0
  2. 调用回溯函数:dfs(nums, 0),从数组的第 0 个位置开始递归。
  3. 返回最终结果 ret

如图分析:

在这里插入图片描述

✨代码

class Solution {
public:
    int ret = 0;
    int path = 0;
    int subsetXORSum(vector<int>& nums) {
        dfs(nums, 0);
        return ret;
    }
    void dfs(vector<int>& nums, int pos){
        ret += path;
        for(int i = pos; i < nums.size(); i++){
            path ^= nums[i];
            dfs(nums, i + 1);
            // 回溯
            path ^= nums[i];
        }
    }
};

✨时间与空间复杂度

🎁时间复杂度
  1. 子集数量:对于数组长度为 n n n,共有 2 n 2^n 2n 个子集。
  2. 异或计算:每次递归执行异或操作的时间复杂度为 O ( 1 ) O(1) O(1)
  3. 总复杂度 O ( 2 n ) O(2^n) O(2n)
🎁空间复杂度
  1. 递归深度:递归调用栈的最大深度为 n n n
  2. 路径变量path 是一个整数,占用 O ( 1 ) O(1) O(1) 的空间。
  3. 总复杂度 O ( n ) O(n) O(n)

🎄四、全排列II

题目链接:https://leetcode.cn/problems/permutations-ii/description/

✨解法简介

利用 回溯法 (Backtracking) 生成所有排列,通过排序和剪枝(跳过重复元素)来避免生成重复排列。


✨步骤详解

🎁Step 1: 初始化
  1. 变量说明
    • ret:存储所有结果的二维数组。
    • path:当前路径,用于构造一个排列。
    • check:布尔数组,记录每个元素是否已经被使用过。
  2. 预处理
    • 对输入数组 nums 进行排序:sort(nums.begin(), nums.end())
      • 排序的目的是将相同的元素放在一起,方便后续剪枝操作。

🎁Step 2: 回溯递归函数设计

递归函数 dfs(nums, pos)

  1. 递归终止条件
    • 如果当前路径 path 的大小等于数组 nums 的长度,说明构造了一组完整排列,将其加入结果集 ret
  2. 循环遍历每个元素
    • 遍历 nums 中的每个元素,尝试将未被使用的元素加入当前路径。
  3. 剪枝条件
    • 元素未被使用:通过 check[i] == false 判断当前元素是否可用。
    • 跳过重复元素:
      • nums[i] == nums[i - 1]:当前元素与前一个元素相同。
      • check[i - 1] == false:前一个相同的元素尚未被使用,说明该排列已经处理过,直接跳过当前分支。
  4. 递归和回溯
    • 递归调用:
      • 将当前元素加入路径,标记为已使用。
      • 递归处理下一个位置。
    • 回溯操作:
      • 递归返回后,将当前元素移出路径,标记为未使用,恢复状态。

🎁Step 3: 主函数调用

在主函数 permuteUnique 中:

  1. 对数组排序:sort(nums.begin(), nums.end())
  2. 初始化布尔数组:check 的大小为 nums.size(),初始值为 false
  3. 调用回溯函数:dfs(nums, 0)
  4. 返回结果集 ret

如图分析:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

✨代码

class Solution {
public:
    vector<vector<int>> ret;
    vector<int> path;
    bool check[9];
    vector<vector<int>> permuteUnique(vector<int>& nums) {
        // 先排序
        sort(nums.begin(), nums.end());
        dfs(nums, 0);
        return ret;
    }
    void dfs(vector<int>& nums, int pos) {
        if (path.size() == nums.size()) {
            ret.push_back(path);
            return;
        }
        for (int i = 0; i < nums.size(); i++) {
            // 剪枝(只关心合法分支)
            if (check[i] == false && (i == 0 || nums[i] != nums[i - 1] || check[i - 1] == true)) {
                path.push_back(nums[i]);
                check[i] = true;
                dfs(nums, pos + 1);
                // 回溯
                path.pop_back();
                check[i] = false;
            }
        }
    }

✨时间与空间复杂度

🎁时间复杂度
  1. 全排列生成:
    • 数组长度为 n n n,排列数量为 n ! n! n!
  2. 剪枝优化:
    • 剪枝减少重复分支,但在最坏情况下仍需要生成 n ! n! n! 个排列。

总时间复杂度 O ( n ⋅ n ! ) O(n \cdot n!) O(nn!)

🎁空间复杂度
  1. 递归深度:
    • 递归调用栈的深度为 O ( n ) O(n) O(n)
  2. 路径和标记数组:
    • 路径数组 path 和布尔数组 check 的大小为 O ( n ) O(n) O(n)

总空间复杂度 O ( n ) O(n) O(n)


结语

回溯法与深度优先搜索(DFS)结合,能有效地解决多种组合问题。通过决策树的结构,回溯法逐步探索解空间,而剪枝则通过减少不必要的计算,显著提高算法的效率。针对特定问题的剪枝优化,可以进一步提升回溯法的性能。

希望通过本文的讲解,您能深入理解回溯法与DFS在解决组合问题中的应用,并通过剪枝技术优化算法效率。如果您对回溯算法或其他算法问题有任何疑问,欢迎交流与讨论!

在这里插入图片描述

今天的分享到这里就结束啦!如果觉得文章还不错的话,可以三连支持一下,17的主页还有很多有趣的文章,欢迎小伙伴们前去点评,您的支持就是17前进的动力!

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/935263.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

STM32--中断

中断 中断向量表 定义一段固定的内存&#xff0c;以4字节对齐&#xff0c;存放各个中断服务函数程序的首地址。定义在启动文件中。 中断相关寄存器 内核中断不经过中断使能、除能寄存器。 中断优先级 1、抢占优先级&#xff1a;高高抢占优先级可以打断正在执行的低抢占优先…

AUTOSAR 汽车开放系统架构

AUTOSAR 官网 AUTOMOTIVE OPEN SYSTEM ARCHITECTURE AUTOSAR (AUTomotive Open System ARchitecture) is a global partnership of leading companies in the automotive and software industry to develop and establish the standardized software framework and open E/E …

《计算机视觉:瓶颈之辩与未来之路》

一、计算机视觉的崛起 计算机视觉是使用计算机模仿人类视觉系统的科学&#xff0c;让计算机拥有类似人类提取、处理、理解和分析图像以及图像序列的能力。它是一个多学科交叉的领域&#xff0c;与机器视觉、图像处理、人工智能、机器学习等领域密切相关。 计算机视觉行业可分为…

Vue 集成地图

电子地图应用广泛&#xff1a; 网约车 : 在网约车 场景中实现 准定位 、导航 、司乘同显 &#xff0c;精准计费 智慧物流、生活服务等&#xff0c;本专题课程囊括各类应用场景 学习 电子地图解决方案&#xff0c;满足学员工作学习各类需求。 基础知识 学习 集成 地图之前需…

Docker Compose实战三:轻松部署PHP

通过前面的文章&#xff08;Docker Compose基础语法与MySQL部署&#xff09;&#xff0c;你已经掌握了Docker Compose的基本语法和常用指令&#xff0c;并成功部署了一个MySQL数据库服务器。今天&#xff0c;我们将继续深入探索Docker Compose的强大功能&#xff0c;介绍如何使…

【深度学习】深刻理解“变形金刚”——Transformer

Transformer 是一种用于处理序列数据的深度学习模型架构&#xff0c;最初由 Vaswani 等人在 2017 年的论文《Attention is All You Need》中提出。它彻底改变了自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;成为许多高级任务&#xff08;如机器翻译、文本生成、问答…

基于springboot+大数据的校园数字图书馆系统

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了多年的设计程序开发&#xff0c;开发过上千套设计程序&#xff0c;没有什么华丽的语言&#xff0c;只有实…

Redis篇-9--数据结构篇1--五种基本结构(String,List,Set,Sorted Set,Hash,BLPOP阻塞逻辑)

Redis 是一个高性能的键值存储系统&#xff0c;支持多种数据结构。每种数据结构都有其独特的特点和适用场景。 1、String&#xff08;字符串&#xff09; &#xff08;1&#xff09;、特点 最简单的数据类型&#xff1a;字符串是最基本的数据类型&#xff0c;可以存储字符串…

优雅的@ObservedV2和@Trace装饰器

Hello&#xff0c;大家好&#xff0c;我是 V 哥。在HarmonyOS NEXT开发中&#xff0c;ObservedV2装饰器和Trace装饰器是用于状态管理的两个装饰器&#xff0c;它们在HarmonyOS应用开发中用于增强对类对象中属性的观测能力。如果你学过观察者模式的原理&#xff0c;你会更容易理…

物联网安全-ARMv8-M Trustzone 实操

前言 本文针对ARMv8m架构M23/M33 MCU安全特性使用进行介绍,以nxp LPC55xx系列和STM32L5xx系列为例,为大家阐述如何使用Trustzone技术提高物联网设备安全性,适合有一定平台安全基础的物联网设备开发人员、安全方案开发人员。 背景 为了提升平台安全性,ARM推出了ARMv8m架构…

昱感微“多维像素”多模态融合感知展示

昱感微采用“多维像素”多模态融合技术&#xff0c;将可见光摄像头、红外摄像头、4D毫米波雷达/激光雷达的探测数据以“多维像素”的数据格式输出&#xff1a;图像数据雷达探测数据红外传感器探测数据叠加&#xff0c;以摄像头像素为颗粒度组合全部感知数据&#xff0c;形成多模…

Launcher添加hotseat图标布局

Launcher的hotseat客户要求添加一些指定应用图标。 首先打开机器将要布局的图标手动移动到hotseat位置上面。 然后使用adb命令将data/data/com.android.launcher3/databases这个文件pull出来。这个文件夹是Luancher的数据库文件。里面保存了相关应用的图标信息。 使用SQLiteS…

GNSS误差源及差分定位

GNSS误差源&#xff1a; &#xff08;一&#xff09;卫星星历误差 由星历信息所得出的卫星位置坐标与实际位置坐标的偏差就是星历误差。星历信息是由 GPS 地面部分测量计算后传入空间部分的。由于卫星在运动中要受到各种摄动力的作用, 而地面部分又很难精确测量这些作用力,…

【数据结构——内排序】希尔排序(头歌实践教学平台习题)【合集】

目录&#x1f60b; 任务描述 测试说明 我的通关代码: 测试结果&#xff1a; 任务描述 本关任务&#xff1a;实现希尔排序算法。 测试说明 平台会对你编写的代码进行测试&#xff1a; 测试输入示例&#xff1a; 10 9 8 7 6 5 4 3 2 1 0 (说明&#xff1a;第一行是元素个数&a…

通俗易懂的 Nginx 反向代理 配置

通俗易懂的 Nginx 反向代理 配置 首先 root 与 alias 的区别 root 是直接拼接 root location location /i/ {root /data/w3; }当请求 /i/top.gif &#xff0c;/data/w3/i/top.gif 会被返回。 alias 是用 alias 替换 location location /i/ {alias /data/w3/images/; }当请…

网页爬虫技术全解析:从基础到实战

引言 在当今信息爆炸的时代&#xff0c;互联网上的数据量每天都在以惊人的速度增长。网页爬虫&#xff08;Web Scraping&#xff09;&#xff0c;作为数据采集的重要手段之一&#xff0c;已经成为数据科学家、研究人员和开发者不可或缺的工具。本文将全面解析网页爬虫技术&…

分页查询和事务管理

前端需要给后端传递的参数&#xff1a; page&#xff1a;当前页码&#xff0c;用于指定用户想要查看的页。pageSize&#xff1a;每页展示记录数&#xff0c;用于指定每页应显示多少条记录。 后端需要给前端返回的结果&#xff1a; total&#xff1a;总记录数&#xff0c;用于告…

MATLAB深度学习(七)——ResNet残差网络

一、ResNet网络 ResNet是深度残差网络的简称。其核心思想就是在&#xff0c;每两个网络层之间加入一个残差连接&#xff0c;缓解深层网络中的梯度消失问题 二、残差结构 在多层神经网络模型里&#xff0c;设想一个包含诺干层自网络&#xff0c;子网络的函数用H(x)来表示&#x…

go语言zero框架调用自己的安装的redis服务配置与使用

在 Go 语言中调用自己安装的 Redis 服务&#xff0c;可以分为几个步骤&#xff1a;从安装 Redis 服务到配置、启动 Redis&#xff0c;最后在 Go 代码中连接并使用 Redis。以下是详细的步骤&#xff1a; ## 1. 安装 Redis 服务 ### 1.1 在 Linux 系统上安装 Redis 假设你使用…

Cerebras 推出 CePO,填补推理与规划能力的关键空白

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…