【算法】C++深度优先搜索(DFS)全解析

📢博客主页:https://blog.csdn.net/2301_779549673
📢欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!
📢本文由 JohnKi 原创,首发于 CSDN🙉
📢未来很长,值得我们全力奔赴更美好的生活✨

在这里插入图片描述

在这里插入图片描述

文章目录

  • 🏳️‍🌈一、DFS 的基础概念
  • 🏳️‍🌈二、DFS 的实现方式![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/438eeca6ac484931aa08b1acb54129d2.png =600x)
    • ❤️(一)递归实现
    • 🧡(二)迭代实现
  • 🏳️‍🌈三、DFS 的应用场景
    • ❤️(一)路径发现
    • 🧡(二)拓扑排序
    • 💛(三)解决谜题和游戏
    • 💚(四)连通性检测
    • 💙(五)生成迷宫
  • 🏳️‍🌈四、DFS 的具体应用示例
    • ❤️(一)迷宫问题
    • 🧡(二)岛屿问题
    • 💛(三)朋友圈问题
    • 💚(四)大洋流水问题
    • 💙(五)图的遍历
  • 👥总结


🏳️‍🌈一、DFS 的基础概念

深度优先搜索(DFS)是一种用于遍历或搜索树或图数据结构的算法。其核心原理是从起始顶点开始,沿着路径尽可能深地探索,直到无法继续前进时才回溯。

DFS 的工作方式类似于在迷宫中探索,一旦进入一个通道,就尽可能地沿着这个通道走到底,直到遇到死胡同或者已经没有未访问的分支,然后再回溯到上一个岔路口,尝试其他路径。形象地说,DFS 就像是一个勇敢的探险家,不断深入未知领域,只有在走投无路时才会回头寻找其他可能性。

在 C++ 中,DFS 可以通过递归或者显式栈来实现。例如,在解决迷宫问题时,我们可以用一个二维数组模拟平面,用一个数组记录坐标是否被访问过,然后从起点开始,使用 DFS 算法不断尝试向四个方向移动,直到找到终点或者遍历完所有可能的路径。

在图的遍历中,DFS 从一个起始节点开始,标记该节点为已访问,然后遍历该节点的所有邻居。如果邻居未被访问,则继续对该邻居进行深度优先搜索。这种方式可以确保图中的每个节点都被访问到,并且可以用于检测图的连通性、寻找路径等问题。
例如,对于一个有向无环图,DFS 可以用于拓扑排序,确定节点的线性排序。在树的遍历中,DFS 可以用于前序、中序、后序遍历,帮助我们更好地理解树的结构和性质。

总之,DFS 是一种强大的算法,在 C++ 编程中有着广泛的应用,可以帮助我们解决各种复杂的问题。

🏳️‍🌈二、DFS 的实现方式在这里插入图片描述

❤️(一)递归实现

在 C++ 中,使用递归实现 DFS 具有简洁易懂的特点。递归实现利用函数调用栈隐式地进行回溯,代码结构清晰,易于理解和实现。以图的遍历为例,递归函数接受当前节点、图的邻接表和访问标记数组为参数,首先标记当前节点为已访问,然后遍历当前节点的所有邻居节点。如果邻居节点尚未访问,则递归地进行深度优先搜索。这种方式使得代码逻辑直观,对于小型图或者树的遍历非常高效。

然而,在遍历大图时,递归实现可能会导致栈溢出问题。这是因为递归调用会在栈上分配空间,当递归深度过大时,栈空间可能会耗尽。例如,在处理非常深的树或者大规模的图时,递归实现可能会因为栈溢出而失败。据统计,在一些实际应用中,当图的节点数超过一定数量(如几万甚至更多)时,递归实现的 DFS 就有可能出现栈溢出问题。

🧡(二)迭代实现

使用栈进行迭代实现 DFS 可以避免在深度较大的图中出现栈溢出风险。迭代实现使用显式栈来模拟递归的过程,每次从栈中取出一个节点,访问其邻居,并将未访问的邻居压入栈。这种方式不依赖于函数调用栈,因此可以处理深度较大的图而不会出现栈溢出问题。

在迭代实现中,我们首先创建一个栈和一个访问标记数组。将起始节点压入栈中,并标记为已访问。然后,在循环中,不断从栈中取出节点进行访问。如果取出的节点尚未访问,则访问该节点,并将其邻居节点压入栈中。通过这种方式,我们可以确保图中的每个节点都被访问到。

与递归实现相比,迭代实现的代码较为冗长,但更加健壮。它可以处理大规模的图,并且可以更好地控制遍历的过程。例如,在处理复杂的图结构时,我们可以根据需要对栈的操作进行优化,以提高遍历的效率。

🏳️‍🌈三、DFS 的应用场景

❤️(一)路径发现

在迷宫问题中,DFS 可以用于寻找从起点到终点的路径。例如,在一个二维迷宫中,我们可以将迷宫看作一个图,每个格子是一个节点,相邻的格子之间有边连接。从起点开始,使用 DFS 不断尝试向四个方向移动,直到找到终点或者遍历完所有可能的路径。在这个过程中,我们可以记录下路径上的节点,以便在找到终点后回溯得到完整的路径。据实际测试,对于一个中等规模的迷宫(如 10x10 的迷宫),DFS 可以在较短的时间内找到路径。

🧡(二)拓扑排序

在有向无环图中,DFS 可以用于拓扑排序。拓扑排序的目标是将图中的节点按照依赖关系进行排序,使得对于任意一条有向边 (u, v),节点 u 在排序后的序列中出现在节点 v 之前。通过对图进行 DFS,并在回溯时将节点加入到排序结果中,可以得到一个满足拓扑顺序的序列。例如,在任务调度问题中,每个任务可能依赖于其他任务的完成,使用拓扑排序可以确定任务的执行顺序,确保所有任务能够按照正确的顺序完成。

💛(三)解决谜题和游戏

在一些谜题和游戏中,DFS 也有广泛的应用。比如八皇后问题,要求在一个 8x8 的棋盘上放置 8 个皇后,使得它们不能相互攻击。可以使用 DFS 遍历所有可能的放置方案,找到满足条件的解。在这个过程中,每次尝试在一个位置放置皇后,如果放置后不满足条件,则回溯到上一步重新尝试。据统计,通过 DFS 可以在合理的时间内找到八皇后问题的所有 92 组解。

💚(四)连通性检测

对于一个图,DFS 可以用于检测其连通性。从图中的任意一个节点开始进行 DFS,如果能够访问到图中的所有节点,则说明该图是连通的;否则,图是不连通的。例如,在网络拓扑结构中,我们可以使用 DFS 来检测网络中的节点是否都能够相互通信。如果网络是连通的,那么从任何一个节点发送的数据都可以到达其他所有节点;如果网络不连通,则需要采取相应的措施来确保数据的传输。

💙(五)生成迷宫

DFS 可以用于生成随机迷宫。一种常见的方法是从一个随机的起点开始,不断地随机选择一个未访问的相邻节点进行扩展,直到无法继续扩展为止。在这个过程中,我们可以记录下路径,从而生成一个迷宫。例如,使用 DFS 生成一个 10x10 的迷宫,通常可以在几毫秒内完成。生成的迷宫具有随机性和复杂性,可以用于游戏开发或者算法测试。

🏳️‍🌈四、DFS 的具体应用示例

❤️(一)迷宫问题

在解决迷宫问题时,我们可以使用 DFS 来寻找从起点到终点的路径。具体实现可以通过递归或者迭代的方式进行。以递归方式为例,我们从起点开始,检查当前位置的四个方向(上、下、左、右)是否可行。如果可行,则继续向该方向进行深度优先搜索,直到找到终点或者无法继续前进。在搜索过程中,我们可以使用一个二维数组来记录已经访问过的位置,以避免重复访问。同时,我们可以使用一个栈或者向量来保存探索过程中的坐标,以便在找到终点后回溯得到完整的路径。

例如,以下是使用递归方式解决迷宫问题的 C++ 代码示例:

#include<iostream>
#include<vector>
const int N = 10; // 迷宫尺寸
int maze[N][N] = { // 迷宫地图,1表示墙,0表示通道
{1,1, 1, 1, 1, 1, 1, 1, 1, 1},
{1,0, 0, 0, 1, 0, 0, 0, 0, 1},
{1,0, 1, 0, 1, 0, 1, 1, 0, 1},
{1,0, 0, 1, 0, 0, 0, 0, 0, 1},
{1,0, 1, 0, 0, 1, 0, 1, 0, 1},
{1,0, 0, 0, 1, 0, 0, 0, 0, 1},
{1,0, 1, 0, 1, 0, 1, 0, 1, 1},
{1,0, 0, 0, 0, 0, 0, 0, 0, 1},
{1,0, 1, 0, 1, 0, 1, 1, 0, 1},
{1,1, 1, 1, 1, 1, 1, 1, 1, 1}
};
std::vector<std::pair<int, int>> path; // 存储路径
bool dfs(int x, int y){ // 深度优先搜索if (x <0 || x >= N || y < 0 || y >= N) return false; // 超出边界if (maze[x][y] ==1) return false; // 遇到墙if (x == N -1 && y == N - 1) { // 到达终点path.emplace_back(x, y);return true;}maze[x][y] = 1; // 标记已访问path.emplace_back(x, y); // 存储路径if (dfs(x+1, y) || dfs(x-1, y) || dfs(x, y+1) || dfs(x, y-1)) { // 沿四个方向搜索return true;}path.pop_back(); // 回溯return false;
}
int main(){dfs(1,1); // 从起点开始搜索if (path.empty()) {std::cout << "No path found." << std::endl;} else {std::cout << "Path:" << std::endl;for (const auto& p : path) {std::cout << "(\" << p.first << \", \" << p.second << \")\" << std::endl;}}return 0;
}

这段代码中,dfs函数使用递归的方式在迷宫中进行深度优先搜索,找到从起点到终点的路径。如果找到了路径,则将路径中的坐标存储在path向量中,并在主函数中输出路径。如果没有找到路径,则输出 “No path found.”。

🧡(二)岛屿问题

以最大岛屿面积问题为例,我们可以使用 DFS 在二维矩阵中解决这个问题。具体来说,我们从一个陆地格子开始,使用 DFS 遍历与其相邻的陆地格子,并记录遍历过的格子数量,即为该岛屿的面积。然后,我们继续遍历其他未被访问过的陆地格子,找到最大的岛屿面积。

以下是使用 DFS 解决最大岛屿面积问题的 C++ 代码示例:

public int maxAreaOfIsland(int[][] grid) {int max = 0;for (int i = 0; i < grid.length; i++) {for (int j = 0; j < grid[0].length; j++) {if (grid[i][j]==1) {max = Math.max(dfs(i, j, grid), max);}}}return max;
}
public int dfs(int i,int j,int[][] grid) {if (i<0||i>=grid.length||j<0||j>=grid[0].length||grid[i][j]==0) {return 0;}grid[i][j]=0;int count=1;count+=dfs(i+1, j, grid);count+=dfs(i-1, j, grid);count+=dfs(i, j+1, grid);count+=dfs(i, j-1, grid);return count;
}

在这段代码中,maxAreaOfIsland函数遍历二维矩阵,找到所有的陆地格子,并调用dfs函数计算每个岛屿的面积。dfs函数使用递归的方式遍历与当前格子相邻的陆地格子,并将其标记为已访问,最后返回岛屿的面积

另外,我们还可以使用方向数组的方式来实现 DFS,代码如下:

int[][] dirs = new int[][]{{-1,0}, {1,0}, {0,-1}, {0,1}};
void dfs(int[][] grid, int i, int j, boolean[] visited) {int m = grid.length, n = grid[0].length;if (i <0 || j < 0 || i >= m || j >= n) {return;}if (visited[i][j]) {return;}visited[i][j] = true;for (int[] d : dirs) {int next_i = i + d[0];int next_j = j + d[1];dfs(grid, next_i, next_j, visited);}
}

这种写法使用方向数组来处理上下左右的遍历,更加简洁和方便。

💛(三)朋友圈问题

在求解朋友圈数量问题中,我们可以使用 DFS 来体现其在传递关系问题中的作用。例如,给定一个 N * N 的矩阵 M,表示班级中学生之间的朋友关系。如果 M [i][j] = 1,表示已知第 i 个和 j 个学生互为朋友关系,否则为不知道。我们可以使用 DFS 算法来计算所有学生中的已知的朋友圈总数。

以下是使用 DFS 解决朋友圈问题的 C++ 代码示例:

class Solution{
public:int findCircleNum(vector<vector<int>>& M) {int n=M.size(),cnt=0;vector<bool>vis(n,false);for(int i=0;i<n;i++){if(!vis[i]){dfs(M,vis,i);++cnt;}}return cnt;}void dfs(vector<vector<int>>& M,vector<bool>& vis,int u){vis[u]=true;int m=M[u].size();for(int i=0;i<m;i++){if(M[u][i]==1&&!vis[i])dfs(M,vis,i);}}
};

在这段代码中,findCircleNum函数遍历所有学生,如果当前学生未被访问,则调用dfs函数进行深度优先搜索,将与当前学生互为朋友的学生标记为已访问,并增加朋友圈数量。dfs函数使用递归的方式遍历与当前学生互为朋友的学生,并将其标记为已访问。

💚(四)大洋流水问题

在大洋流水问题中,我们可以从大洋开始向上流,利用 DFS 确定能流到太平洋和大西洋的位置。具体来说,我们从太平洋和大西洋的边界开始,使用 DFS 遍历与其相邻的格子,如果格子的高度不小于当前格子的高度,则可以流向该格子。最后,我们遍历整个矩阵,找到既能流到太平洋又能流到大西洋的格子。

以下是使用 DFS 解决大洋流水问题的 C++ 代码示例:

class Solution{
public:List<List<Integer>> pacificAtlantic(int[][] matrix) {List<List<Integer>> res = new ArrayList<>();if (matrix.length == 0 || matrix[0].length == 0) {return res;}int r = matrix.length; //行数,可以看成 y 轴int c = matrix[0].length; //列数,可以看成 x 轴boolean[][] toPa = new boolean[r][c];boolean[][] toAt = new boolean[r][c];for (int i = 0; i < r; i++) {DFS(i, 0, matrix, toPa, Integer.MIN_VALUE); //遍历第一列,即正方形的左边,它用 DFS 来遍历哪些能够通过它,到达太平洋(Pa)的水流。DFS(i, c - 1, matrix, toAt, Integer.MIN_VALUE); //遍历最后一列}for (int i = 0; i < c; i++) {DFS(0, i, matrix, toPa, Integer.MIN_VALUE); //遍历第一行DFS(r - 1, i, matrix, toAt, Integer.MIN_VALUE); //遍历最后一行}for (int i = 0; i < r; i++) {for (int j = 0; j < c; j++) {if (toPa[i][j] && toAt[i][j]) {List<Integer> cur = new ArrayList<>();cur.add(i);cur.add(j);res.add(cur);}}}return res;}int[][] directions = new int[][]{{0,1},{0,-1},{-1,0},{1,0}};public void DFS(int r, int c, int[][] matrix, boolean[][] toSea, int height) {if (r < 0 || r >= matrix.length || c < 0 || c >= matrix[0].length || toSea[r][c] || matrix[r][c]< height) {return;}toSea[r][c]=true;for (int[] dir : directions) {DFS(r + dir[0], c + dir[1], matrix, toSea, matrix[r][c]);}}
};

在这段代码中,pacificAtlantic函数首先初始化两个布尔数组toPa和toAt,分别表示能够流到太平洋和大西洋的格子。然后,从太平洋和大西洋的边界开始,使用 DFS 遍历与其相邻的格子,并将能够流到相应海洋的格子标记为true。最后,遍历整个矩阵,找到既能流到太平洋又能流到大西洋的格子,并将其坐标加入结果列表中。DFS函数使用递归的方式遍历与当前格子相邻的格子,并将能够流到相应海洋的格子标记为true。

💙(五)图的遍历

以无向图和二叉树的遍历为例,展示 DFS 在不同数据结构中的具体实现。
在无向图的遍历中,我们可以使用 DFS 来遍历图中的所有节点。具体实现可以通过递归或者迭代的方式进行。以递归方式为例,我们从一个起始节点开始,标记该节点为已访问,然后遍历该节点的所有邻居节点。如果邻居节点尚未访问,则递归地进行深度优先搜索。

以下是使用递归方式遍历无向图的 C++ 代码示例:

class undigraph {
public:void DFS(int v) {cout << "顶点" << v << "被访问!" << endl;visited[v] = true;for (int prev = smallestadjvertex(v); prev >= 0; prev = nextadjvertex(v, prev)) {if (visited[prev] == false) {DFS(prev);}}}int smallestadjvertex(int v) {for (int i = 0; i < vertexnum; i++) {if (adjmatrix[v][i] == 1) {return i;}return -1;}}int nextadjvertex(int v, int prev) {for (int i = prev + 1; i < vertexnum; i++) {if (adjmatrix[v][i] == 1) {return i;}return -1;}}void DFStraverse() {for (int i = 0; i < vertexnum; i++) {visited[i] = false;}for (int i = 0; i < vertexnum; i++) {if (visited[i] == false) {DFS(i);}}}
private:bool visited[maxsize];int adjmatrix[maxsize][maxsize];int vertexnum;
};

在这段代码中,undigraph类表示无向图,其中DFS函数使用递归的方式遍历无向图中的节点,smallestadjvertex函数和nextadjvertex函数分别用于寻找当前节点的最小序号邻接顶点和下一个邻接顶点,DFStraverse函数用于遍历整个无向图。

在二叉树的遍历中,DFS 可以用于前序、中序和后序遍历。以下是使用递归方式实现二叉树的前序、中序和后序遍历的 C++ 代码示例:

struct TreeNode {int val;TreeNode *left;TreeNode *right;TreeNode(int x) : val(x), left(NULL), right(NULL) {}
};
void preOrderRecur(TreeNode *root) {if (!root) return;cout << root->val << " ";preOrderRecur(root->left);preOrderRecur(root->right);
}
void inOrderRecur(TreeNode *root) {if (!root) return;inOrderRecur(root->left);cout << root->val << " ";inOrderRecur(root->right);
}
void postOrderRecur(TreeNode *root) {if (!root) return;postOrderRecur(root->left);postOrderRecur(root->right);cout << root->val << " ";
}

在这段代码中,TreeNode结构体表示二叉树的节点,preOrderRecur函数、inOrderRecur函数和postOrderRecur函数分别用于实现二叉树的前序、中序和后序遍历。这些函数使用递归的方式遍历二叉树中的节点,并输出节点的值。

总之,DFS 在不同的数据结构中有着广泛的应用,可以帮助我们解决各种复杂的问题。


👥总结


本篇博文对 深度优先搜索(DFS) 做了一个较为详细的介绍,不知道对你有没有帮助呢

觉得博主写得还不错的三连支持下吧!会继续努力的~

请添加图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

汽车免拆诊断案例 | 2010款起亚赛拉图车发动机转速表指针不动

故障现象  一辆2010款起亚赛拉图车&#xff0c;搭载G4ED 发动机&#xff0c;累计行驶里程约为17.2万km。车主反映&#xff0c;车辆行驶正常&#xff0c;但组合仪表上的发动机转速表指针始终不动。 故障诊断  接车后进行路试&#xff0c;车速表、燃油存量表及发动机冷却温度…

【环境搭建】Apache ZooKeeper 3.8.4 Stable

软件环境 Ubuntu 20.04 、OpenJDK 11 OpenJDK 11&#xff08;如果已经安装&#xff0c;可以跳过这一步&#xff09; 安装OpenJDK 11&#xff1a; $ sudo apt-get update$ sudo apt-get install -y openjdk-11-jdk 设置 JAVA_HOME 环境变量&#xff1a; $ sudo gedit ~/.bash…

solid works下载

软件安装包下载解压打开 将软件安装包下载到电脑本地&#xff0c;使用解压工具进行解压打开&#xff08;下载解压安装全程关闭杀毒软件及防火墙&#xff09; 打开Crack文件夹 打开Crack文件夹进去 复制SolidWorks_Flexnet_Server文件夹 复制SolidWorks_Flexnet_Server文件夹到…

硅谷甄选(10)用户管理

用户管理模块 9.1 静态搭建 主要是el-form、el-pagination <template><el-card style"height: 80px"><el-form :inline"true" class"form"><el-form-item label"用户名:"><el-input placeholder"请…

C语言-详细讲解-打印各种三角形矩阵

目录 1.字符三角形 2. 直角三角形&#xff08;1&#xff09; 3.直角三角形&#xff08;2&#xff09; 4. 打印乘法口诀表 1.字符三角形 入门级&#xff0c;熟悉输入与输出 #include<stdio.h> int main() {char i;scanf("%c",&i);printf(" %c…

图像的特征类别

一 图像的特征类别 1.自然特征 ①光谱特征 ②几何特征 ③时相特征 2.人工特征 ①直方图特征 ②灰度边缘特征 ③线、角点、纹理特征 图像的特征有很多&#xff0c;按提取特征的范围大小又可分为&#xff1a; ①点特征 仅由各个像素就能决定的性质。如单色图像中的灰度值、彩色…

MYSQL--完整性约束

目录 约束 设置约束 方式一 推荐&#xff1a;方式二 删除约束 主键约束 自增列 组合主键, 联合主键 外键约束 给表添加外键约束 注意: 约束 概念&#xff1a;约束是作用于表中字段上的规则&#xff0c;用于限制存储在表中的数据。 目的&#xff1a;保证数据库中数据的正…

线程池面试点

为什么使用线程池&#xff1f;线程池的作用是什么&#xff1f;线程池的执行顺序&#xff1f;线程池的入参是哪些&#xff1f;分别代表什么意思&#xff1f; 频繁的创建线程与销毁线程,资源耗费与上下文切换时间开销非常大,而线程池可以避免这种情况. 入参有核心线程数,最大线…

IDEA安装(AI)MarsCode插件

1. 注册MarsCode点击注册MarsCode账号 2. 安装MarsCode 第一步: 打开IDEA设置页面, File -> Settings… 第二步: 在左侧导航栏中选择 “Plugins”, 在 “Marketplace” 中搜索"mars", 如图, 点击 “Installed” 第三步: 插件安装完成后, 重启IDEA. 点击右侧导航栏…

软考高项比中项在难度上高多少?

论文这一科难倒了70%的考生&#xff0c;再加上30%的项目管理技能运用知识&#xff0c;使得考生们感到困难重重。题主提到的高额培训费主要是因为论文这一科目的难度较大。虽然直接考取高级资格是理想状态&#xff0c;但有些人可能无法一次成功&#xff0c;因此可以先考取中级资…

飞腾平台Arm ComputeLibrary编译安装指南

【写在前面】 飞腾开发者平台是基于飞腾自身强大的技术基础和开放能力&#xff0c;聚合行业内优秀资源而打造的。该平台覆盖了操作系统、算法、数据库、安全、平台工具、虚拟化、存储、网络、固件等多个前沿技术领域&#xff0c;包含了应用使能套件、软件仓库、软件支持、软件适…

【数据结构】堆:建堆/向下调整/上向调整/堆排序/TOK问题

文章目录 前言堆的定义1.大小堆2.完全二叉树 堆的实现堆的数据结构初始化销毁取堆顶元素判断堆是否为空父结点和子结点下标关系&#xff08;重要&#xff09; 向下调整法-O(n)小堆版大堆版 向上调整法-nlog(n)堆的插入和删除插入(调用向上调整)删除(调用向下调整) 构建最大堆向…

Qt第三课 ----------输入类的控件属性

作者前言 &#x1f382; ✨✨✨✨✨✨&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f382; ​&#x1f382; 作者介绍&#xff1a; &#x1f382;&#x1f382; &#x1f382; &#x1f389;&#x1f389;&#x1f389…

java面试2.0

一.Zookeeper 1.定义 ZooKeeper 是一个开源的分布式协调服务&#xff0c;它的设计目标是将那些复杂且容易出错的分布式一致性服务封装起来&#xff0c;构成一个高效可靠的原语集&#xff0c;并以一系列简单易用的接口提供给用户使用。 ZooKeeper 为我们提供了高可用、高性能…

【论文复现】语言模型中的多模态链式推理

&#x1f4d5;作者简介&#xff1a;热爱跑步的恒川&#xff0c;致力于C/C、Java、Python等多编程语言&#xff0c;热爱跑步&#xff0c;喜爱音乐、摄影的一位博主。 &#x1f4d7;本文收录于论文复现系列&#xff0c;大家有兴趣的可以看一看 &#x1f4d8;相关专栏C语言初阶、C…

SPA——MVC 与 MVVM - 2024最新版前端秋招面试短期突击面试题【100道】

SPA——MVC 与 MVVM - 2024最新版前端秋招面试短期突击面试题【100道】 &#x1f310; 在现代前端开发中&#xff0c;单页面应用&#xff08;SPA&#xff09;以及不同的架构模式&#xff08;如MVC和MVVM&#xff09;是实现用户交互的关键概念。了解它们的优缺点以及如何应用&a…

前端八股文第五篇

41. 做的比较好的项目&#xff0c;展开讲讲 我曾参与开发过一个电商平台的项目&#xff0c;该项目是一个多端适配的电商平台&#xff0c;包括网页端、移动端和小程序端。在这个项目中&#xff0c;我主要负责前端开发工作。我们团队采用了 Vue.js 框架进行开发&#xff0c;使用…

运筹说 第127期 | 存储论相关模型代码实现

本期我们进行运筹学之存储论算法的讲解&#xff0c;我们将对常见的商品存储模型进行介绍&#xff0c;并求解不同供需条件下的最优存储策略&#xff0c;通过MATLAB和Python工具帮助大家快速求解最优存储策略&#xff0c;做到事半功倍。由于篇幅有限&#xff0c;小编接下来只展示…

【Linux】Linux权限

本文将会围绕着下面5个问题进行讲解&#xff1a; 认识Linux下用户的分类什么叫做权限没有权限会有什么现象如何进行权限的修改其他与权限相关的问题 好了&#xff0c;本章内容就要开始了&#xff0c;准备你的大脑&#xff0c;开凿吧&#xff01;&#xff01;&#xff01; 认识…

Ubuntu系统如何实现键盘按键映射到其他按键(以 Ctrl+c 映射到 F3,Ctrl+v 映射到 F4 为例)

文章目录 写在前面1. 功能描述2. 实现步骤2.1 安装AutoKey2.2 软件设置2.2.1 软件设置 2.3 测试是否安装成功 参考链接 写在前面 自己的测试环境&#xff1a; Ubuntu20.04 1. 功能描述 Ubuntu系统使用Ctrlc 、Ctrlv 进行复制粘贴操作的时候&#xff0c;时间长了就会出现小拇指…