【六十】【算法分析与设计】用一道题目解决dfs深度优先遍历,dfs中节点信息,dfs递归函数模板进入前维护出去前回溯,唯一解的剪枝飞升返回值true

路径之谜

题目描述

小明冒充X星球的骑士,进入了一个奇怪的城堡。

城堡里边什么都没有,只有方形石头铺成的地面。

假设城堡地面是n×n个方格。如下图所示。

1e8908abd7eb470083d000bb49a7bab3.png

按习俗,骑士要从西北角走到东南角。可以横向或纵向移动,但不能斜着音走,也不能跳跃。每走到一个新方格,就要向正北

方和正西方各射一箭。(城堡的西墙和北墙内各有12个靶子)同一个方格只允许经过一次。但不必走完所有的方格。如果只

给出靶子上箭的数目,你能推断出骑士的行走路线吗?有时是可以的,比比如上图中的例子。

本题的要求就是已知箭靶数字,求骑士的行走路径(测试)数据保证路径唯一)

输入描述

第一行一个整数N(0<N<20),表示地面有NXN个方格。

第二行N个整数,空格分开,表示北边的箭靶上的数字(自西向东)

第三行N个整数,空格分开,表示西边的箭靶上的数字(自北向南)

输出描述

输出一行若干个整数,表示骑士路径。

为了方便表示,我们约定每个小格子用一个数字代表,从西北角开始编号号:0,1,2,3

比如,上图中的方块编号为:

0 1 2 3

4 5 6 7

8 9 10 11

12 13 14 15

输入输出样例

示例

输入

4

2 4 3 4

4 3 3 3

输出

0 4 5 1 2 3 7 11 10 9 13 14 15

运行限制

最大运行时间:5s

最大运行内存:256M

12c651f1c7234b9988867f9e74ddcf2e.png

暴力递归

1.

定义dfs(i,j)表示当前节点坐标。

假设入口位置坐标是(1,1),往下是row行,往右是col列。

定义row记录从入口到当前节点这条路径西边靶子的数量,row[1]表示西边第一个靶子上箭的数量,row[2]表示西边第二个靶子上箭的数量...以此类推。

定义col记录从入口到当前节点这条路径北边靶子的数量,col[1]表示北边第一个靶子上箭的数量,col[2]表示北边第二个靶子上箭的数量...以此类推。

定义path记录从入口到当前节点的路径信息。

定义visit记录从入口到当前节点,这条路径,当前所有位置访问情况,访问过为true,没有被访问过false。

2.

也就是当前节点的信息不止有(i,j)还有path,row,col,visit四个变量共同组成。

3.

dfs内部逻辑,进入当前节点的时候,维护当前节点的所有信息。

离开当前节点返回之前,回溯,消除当前节点维护的所有信息。

4.

夹在中间的就是计算逻辑,写代码的时候先把首位维护和回溯写掉,然后在中间加主要逻辑。

 
void dfs(int i, int j) {visit[i][j] = true;row[i]++;col[j]++;path.push_back((i - 1) * n + (j - 1));//数学推导不细说,找规律//添加主要逻辑path.pop_back();row[i]--;col[j]--;visit[i][j] = false;
}

 

5.

06e70e01efad4a72a6044d401aaf9bae.png

对于当前节点,下一个遍历的节点位置有四个,上下左右,但是有一些位置需要剪枝。

规则是,第一,这四个位置首先不能越界,第二,这四个位置不能被访问过,被访问过表示已经是路径上的点,已经走过了。如果满足要求就可以dfs进入下一节点。

 
void dfs(int i, int j) {visit[i][j] = true;row[i]++;col[j]++;path.push_back((i - 1) * n + (j - 1));for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 1 && x <= n && y >= 1 && y <= n && !visit[x][y]) {dfs(x, y);}}path.pop_back();row[i]--;col[j]--;visit[i][j] = false;
}

 

6.

思考递归出口,递归出口是当你走到出口的时候,即i==n&&j==n。

此时遍历row和col,看看靶子上箭的数量是不是和aim_row,aim_col目标值对上。

如果对上了此时path里面存放的就是我们要的路径,打印出来即可。

如果没有对上就返回,不需要再进入下一节点了。

返回前注意需要回溯,消除维护操作。

 
void dfs(int i, int j) {visit[i][j] = true;row[i]++;col[j]++;path.push_back((i - 1) * n + (j - 1));if (i == n && j == n) {int flag = 1;for (int i = 1; i <= n; i++) {if (row[i] != aim_row[i] || col[i] != aim_col[i]) flag = 0;}if (flag == 1) {for (auto& x : path) cout << x << " ";}path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return;}for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 1 && x <= n && y >= 1 && y <= n && !visit[x][y]) {dfs(x, y);}}path.pop_back();row[i]--;col[j]--;visit[i][j] = false;
}

 

代码1

7.

此时得到的代码如下,

 
#include <iostream>
#include<bits/stdc++.h>
using namespace std;int n; // 定义n表示城堡地面是n×n个方格
vector<int> path; // 用于记录骑士的行走路径
vector<int> row; // 用于记录每行射出的箭的数量
vector<int> col; // 用于记录每列射出的箭的数量
vector<vector<int>> visit; // 记录某个格子是否被访问过
int dx[4] = { 1,-1,0,0 }; // 方向数组,用于实现上下左右移动
int dy[4] = { 0,0,1,-1 };
vector<int> aim_col; // 存储每列应该射出的箭的目标数量
vector<int> aim_row; // 存储每行应该射出的箭的目标数量// 深度优先搜索(DFS)函数,用于尝试所有可能的路径
void dfs(int i, int j) {visit[i][j] = true; // 标记当前格子为已访问row[i]++; // 当前行的箭数增加col[j]++; // 当前列的箭数增加path.push_back((i - 1) * n + (j - 1)); // 将当前格子编号加入路径// 如果到达东南角,并且每行每列的箭数都符合目标if (i == n && j == n) {int flag = 1;for (int i = 1; i <= n; i++) {if (row[i] != aim_row[i] || col[i] != aim_col[i]) flag = 0;}if (flag == 1) {for (auto& x : path) cout << x << " "; // 如果路径有效,输出这条路径}path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return;}// 尝试向四个方向移动for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 1 && x <= n && y >= 1 && y <= n && !visit[x][y]) {dfs(x, y);}}// 回溯,撤销当前步骤的影响path.pop_back();row[i]--;col[j]--;visit[i][j] = false;
}int main() {cin >> n; // 读入n的大小aim_col.resize(n + 1);for (int i = 1; i <= n; i++) cin >> aim_col[i]; // 读入每列的目标箭数aim_row.resize(n + 1);for (int i = 1; i <= n; i++) cin >> aim_row[i]; // 读入每行的目标箭数row.resize(n + 1);col.resize(n + 1);visit = vector<vector<int>>(n + 1, vector<int>(n + 1));dfs(1, 1); // 从西北角开始进行深度优先搜索return 0;
}

运行结果如下,

09945d723f644fc8a58e3ff7d6749755.png

剪枝1:唯一解找到即返回true(飞升)

8.

大部分运行超时,说明代码整体逻辑没有问题,但是剪枝操作没有做好。

重新思考剪枝操作。

题目中测试数据保证了路径的唯一性,说明我们只要找到了最终答案,就不需要回溯,后面的操作都不需要,找到最终答案就一路飞升回到第一层递归返回。

修改dfs的返回值,不使用void返回值,而使用int或者bool,意思是如果当前找到了就返回true,没有找到就返回false。

修改完返回值还需要修改进入下一层递归的代码,不能直接是dfs,而是if(dfs(x, y)) return true;

如果返回值是true说明找到了,如果找到了什么都不用管,直接返回true。

每一层节点都接收这个信息,有没有完成工作,有没有找到路径?找到了就可以不用再工作了,直接返回。

还需要修改递归出口的逻辑,flag==1说明找到了,打印完路径后直接返回true。

 
int dfs(int i, int j) {visit[i][j] = true;row[i]++;col[j]++;path.push_back((i - 1) * n + (j - 1));if (i == n && j == n) {int flag = 1;for (int i = 1; i <= n; i++) {if (row[i] != aim_row[i] || col[i] != aim_col[i]) flag = 0;}if (flag == 1) {for (auto& x : path) cout << x << " ";return true;}path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return;}for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 1 && x <= n && y >= 1 && y <= n && !visit[x][y]) {if(dfs(x, y)) return true;}}path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false;
}

 

代码2

9.

 
#include <iostream>
#include<bits/stdc++.h>
using namespace std;int n; // 城堡地面为n×n个方格的大小
vector<int> path; // 存储骑士的行走路径
vector<int> row; // 每行射箭的次数
vector<int> col; // 每列射箭的次数
vector<vector<int>> visit; // 标记方格是否访问过
int dx[4] = { 1,-1,0,0 }; // x方向的移动:下,上,不动,不动
int dy[4] = { 0,0,1,-1 }; // y方向的移动:不动,不动,右,左
vector<int> aim_col; // 目标,每列应射箭的次数
vector<int> aim_row; // 目标,每行应射箭的次数// 深度优先搜索(DFS)算法,i和j表示当前位置
int dfs(int i, int j) {visit[i][j] = true; // 标记当前方格为已访问row[i]++; // 当前行的射箭次数增加col[j]++; // 当前列的射箭次数增加path.push_back((i - 1) * n + (j - 1)); // 将当前方格的编号加入到路径中// 检查是否到达右下角,并且所有行和列的射箭次数都符合目标if (i == n && j == n) {int flag = 1; // 用于检查是否所有行和列的射箭次数都匹配for (int i = 1; i <= n; i++) {if (row[i] != aim_row[i] || col[i] != aim_col[i]) flag = 0;}if (flag == 1) {for (auto& x : path) cout << x << " "; // 如果匹配,输出路径cout << endl; // 输出换行return true; // 返回找到有效路径}}// 尝试四个可能的移动方向for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 1 && x <= n && y >= 1 && y <= n && !visit[x][y]) {if (dfs(x, y)) return true; // 递归调用dfs}}// 回溯,撤销当前步骤path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false; // 没有找到有效路径
}int main() {cin >> n;aim_col.resize(n + 1);for (int i = 1; i <= n; i++) cin >> aim_col[i]; // 输入每列的目标射箭次数aim_row.resize(n + 1);for (int i = 1; i <= n; i++) cin >> aim_row[i]; // 输入每行的目标射箭次数row.resize(n + 1);col.resize(n + 1);visit = vector<vector<int>>(n + 1, vector<int>(n + 1, false)); // 初始化访问标记数组dfs(1, 1); // 从(1,1)开始深度优先搜索return 0;
}

运行结果如下,

a94fedcc6cc74cd2bcb5c4451d3fd699.png

剪枝2:靶子箭数量小于目标靶子箭数量

10.

说明这个剪枝操作微不足道,没什么用。

继续思考其他剪枝操作,我们发现如果到了一个节点,如果row,col中有一个靶子的箭数量大于目标靶子的箭数量,后面的路径都不可能是最终答案。

因为靶子上箭的数量不可能减少只能增加,所以到出口之前,如果是正确路径,靶子数量一定小于目标靶子数量。

所以如果靶子箭数量大于目标靶子箭数量,直接返回。

 
int dfs(int i, int j) {visit[i][j] = true;row[i]++;col[j]++;path.push_back((i - 1) * n + (j - 1));int flag1 = 1;for (int i = 1; i <= n; i++) {if (row[i] > aim_row[i] || col[i] > aim_col[i]) {flag1 = 0;break;}}if (flag1 == 0) {path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false;}if (i == n && j == n) {int flag = 1;for (int i = 1; i <= n; i++) {if (row[i] != aim_row[i] || col[i] != aim_col[i]) {flag = 0;break;}}if (flag == 1) {for (auto& x : path) cout << x << " ";return true;}else{path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false;}}for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 1 && x <= n && y >= 1 && y <= n && !visit[x][y]) {if (dfs(x, y)) return true;}}path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false;
}

 

代码3

 
#include <iostream>
#include<bits/stdc++.h>  // 引入常用库,包含STL等
using namespace std;// 定义全局变量
int n;  // 地图大小
vector<int> path;  // 记录路径
vector<int> row;  // 记录每行走过的次数
vector<int> col;  // 记录每列走过的次数
vector<vector<int>> visit;  // 访问标记数组
int dx[4] = { 1,-1,0,0 };  // 方向数组,表示行的移动
int dy[4] = { 0,0,1,-1 };  // 方向数组,表示列的移动
vector<int> aim_col;  // 目标列的箭数
vector<int> aim_row;  // 目标行的箭数// 深度优先搜索函数
int dfs(int i, int j) {visit[i][j] = true;  // 标记当前单元格已访问row[i]++;  // 增加当前行的计数col[j]++;  // 增加当前列的计数path.push_back((i - 1) * n + (j - 1));  // 记录路径int flag1 = 1;// 检查所有行列是否满足条件for (int i = 1; i <= n; i++) {if (row[i] > aim_row[i] || col[i] > aim_col[i]) {flag1 = 0;break;}}if (flag1 == 0) {  // 如果条件不满足,则回退操作path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false;}// 检查是否到达最后一个方格if (i == n && j == n) {int flag = 1;for (int i = 1; i <= n; i++) {if (row[i] != aim_row[i] || col[i] != aim_col[i]) {flag = 0;break;}}if (flag == 1) {  // 如果满足最终条件,输出路径for (auto& x : path) cout << x << " ";return true;} else {  // 否则,进行回退操作path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false;}}// 遍历四个方向for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 1 && x <= n && y >= 1 && y <= n && !visit[x][y]) {if (dfs(x, y)) return true;}}// 回退操作path.pop_back();row[i]--;col[j]--;visit[i][j] = false;return false;
}int main() {cin >> n;aim_col.resize(n + 1);for (int i = 1; i <= n; i++) cin >> aim_col[i];  // 输入北边靶子箭数aim_row.resize(n + 1);for (int i = 1; i <= n; i++) cin >> aim_row[i];  // 输入西边靶子箭数row.resize(n + 1);col.resize(n + 1);visit = vector<vector<int>>(n + 1, vector<int>(n + 1));  // 初始化访问矩阵dfs(1, 1);  // 从(1,1)开始搜索return 0;
}

72a4030d74204f56b86d822e926b7774.png

总结结论

1.

递归函数dfs,用同样的函数完成相同的逻辑问题,但是需要用其他遍历辅助判断当前位于那个节点。

vector<int> path;

vector<int> row;

vector<int> col;

vector<vector<int>> visit;

(i,j)

以上都是当前节点的信息。

2.

每一次进入dfs维护当前节点信息,每一次出去之前回溯,消除维护的信息。

 
void dfs(int i, int j) {visit[i][j] = true;row[i]++;col[j]++;path.push_back((i - 1) * n + (j - 1));//数学推导不细说,找规律//添加主要逻辑path.pop_back();row[i]--;col[j]--;visit[i][j] = false;
}

 

3.

如果只需要找唯一解,找到即返回,找到就不用工作,找到就飞升。

修改返回值,进入下一层逻辑,出口逻辑。

7401f170df974efcb1dc440636a9fd40.png

4.

剪枝操作提前返回,注意返回之前一定要回溯,也就是消除维护当前节点信息的操作!!!

结尾

最后,感谢您阅读我的文章,希望这些内容能够对您有所启发和帮助。如果您有任何问题或想要分享您的观点,请随时在评论区留言。

 

同时,不要忘记订阅我的博客以获取更多有趣的内容。在未来的文章中,我将继续探讨这个话题的不同方面,为您呈现更多深度和见解。

 

谢谢您的支持,期待与您在下一篇文章中再次相遇!

 

 

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ADOP带您科普什么是单纤双向BiDi光模块?一根光纤,双向通信:单纤双向模块的革命性技术。

单纤双向光模块&#xff08;也称为BiDi光模块&#xff09;是一种使用WDM&#xff08;波分复用&#xff09;双向传输技术的光模块&#xff0c;它在一根光纤上实现了同时进行光通道内的双向传输。相比常规光模块&#xff08;有两个光纤插孔&#xff09;&#xff0c;BiDi光模块只有…

基于Python+Selenium+Pytest的Dockerfile如何写

使用 Dockerfile 部署 Python 应用程序与 Selenium 测试 在本文中&#xff0c;我们将介绍如何使用 Dockerfile 部署一个 Python 应用程序&#xff0c;同时利用 Selenium 进行自动化测试。我们将使用官方的 Python 运行时作为父镜像&#xff0c;并在其中安装所需的依赖项和工具…

【Node.js工程师养成计划】之打造自己的脚手架工具

一、创建全局的自定义命令 1、打开一个空文件夹&#xff0c;新建一个bin文件夹&#xff0c;在bin文件夹下新建cli.js文件&#xff0c;js文件可以命名为cli.js&#xff08;您随意&#xff09; 2、在cli.js文件中的开头&#xff08;&#xff01;&#xff01;&#xff09;写下面这…

windows环境下安装Apache

首先apache官网下载地址&#xff1a;http://www.apachelounge.com/download/按照自己的电脑操作系统来安装 这里我安装的是win64 主版本是2.4的apache。 然后解压压缩包到一个全英文的路径下&#xff01;&#xff01;&#xff01;一定一定不要有中文 中文符号也不要有&#xff…

十一、Yocto集成tcpdump等网络工具

文章目录 Yocto集成tcpdump等网络工具networking layer集成 Yocto集成tcpdump等网络工具 本篇文章为基于raspberrypi 4B单板的yocto实战系列的第十一篇文章&#xff1a; 一、yocto 编译raspberrypi 4B并启动 二、yocto 集成ros2(基于raspberrypi 4B) 三、Yocto创建自定义的lay…

RabbitMQ工作模式(5) - 主题模式

概念 主题模式&#xff08;Topic Exchange&#xff09;是 RabbitMQ 中一种灵活且强大的消息传递模式&#xff0c;它允许生产者根据消息的特定属性将消息发送到一个交换机&#xff0c;并且消费者可以根据自己的需求来接收感兴趣的消息。主题交换机根据消息的路由键和绑定队列的路…

演示在一台Windows主机上运行两个Mysql服务器(端口号3306 和 3307),安装步骤详解

目录 在一台Windows主机上运行两个Mysql服务器&#xff0c;安装步骤详解因为演示需要两个 MySQL 服务器终端&#xff0c;我只有一个 3306 端口号的 MySQL 服务器&#xff0c;所以需要再创建一个 3307 的。创建一个3307端口号的MySQL服务器1、复制 mysql 的安装目录2、修改my.in…

NAT网络地址转换实验(华为)

思科设备参考&#xff1a;NAT网络地址转换实验&#xff08;思科&#xff09; 一&#xff0c;技术简介 NAT&#xff08;Network Address Translation&#xff09;&#xff0c;即网络地址转换技术&#xff0c;是一种在现代计算机网络中广泛应用的技术&#xff0c;主要用于有效管…

nvm基本使用

nvm基本使用 文章目录 nvm基本使用1.基本介绍2.下载地址3.常用指令 1.基本介绍 NVM是一个用于管理 Node.js 版本的工具。它允许您在同一台计算机上同时安装和管理多个 Node.js 版本&#xff0c;针对于不同的项目可能需要不同版本的 Node.js 运行环境。 NVM 主要功能&#xff…

百度智能云千帆 ModelBuilder 技术实践系列:通过 SDK 快速构建并发布垂域模型

​百度智能云千帆大模型平台&#xff08;百度智能云千帆大模型平台 ModelBuilder&#xff09;作为面向企业开发者的一站式大模型开发平台&#xff0c;自上线以来受到了广大开发者、企业的关注。至今已经上线收纳了超过 70 种预置模型服务&#xff0c;用户可以快速的调用&#x…

STM32的端口引脚的复用功能及重映射功能解析

目录 STM32的端口引脚的复用功能及重映射功能解析 复用功能 复用功能的初始化 重映射功能 重映射功能的初始化 复用功能和重映射的区别 部分重映射与完全重映射 补充 STM32的端口引脚的复用功能及重映射功能解析 复用功能 首先、我们可以这样去理解stm32引脚的复用功能…

docker容器技术篇:容器集群管理实战mesos+zookeeper+marathon(一)

容器集群管理实战mesoszookeepermarathon&#xff08;一&#xff09; mesos概述 1.1 Mesos是什么 Apache Mesos 是一个基于多资源调度的集群管理软件&#xff0c;提供了有效的、跨分布式应用或框架的资源隔离和共享&#xff0c;可以运行 Hadoop、Spark以及docker等。 1.2 为…

Flowable 基本用法

一. 什么是Flowable Flowable 是一个基于 Java 的开源工作流引擎&#xff0c;用于实现和管理业务流程。它提供了强大的工作流引擎和一套丰富的工具&#xff0c;使开发人员能够轻松地建模、部署、执行和监控各种类型的业务流程。Flowable 是 Activiti 工作流引擎的一个分支&am…

服务网关GateWay基础

1. 网关基础介绍1.1 网关是什么1.2 为啥要用网关1.3 常见的网关组件NginxNetflix ZuulSpring Cloud GatewayKongAPISIX综合比较 2. gateWay的使用2.1 springCloud整合gateway2.2 GateWay的相关用法2.3 GateWay路由使用示例基本用法转发/重定向负载请求动态路由 2.5 断言(Predic…

使用Screenshots安装Fedora 40版本详细教程

Fedora 40是Fedora操作系统的最新版本&#xff0c;于 2024 年 4 月 23 日发布&#xff0c;是一个社区支持的 Linux 发行版&#xff0c;以其创新功能、领先技术和活跃的社区支持而闻名。 在本指南中&#xff0c;我们将引导您完成安装Fedora 40 Server的分步过程&#xff0c;确保…

Docker之存储配置与管理

一、容器本地配置与Docker存储驱动 每个容器都被自动分配了本地存储&#xff0c;也就是内部存储。容器由一个可写容器层和若干只读镜像层组成&#xff0c;容器的数据就存放在这些层中。 容器本地存储采用的是联合文件系统。这种文件系统将其他文件系统合并到一个联合挂载点&a…

【c++】深入剖析与动手实践:C++中Stack与Queue的艺术

&#x1f525;个人主页&#xff1a;Quitecoder &#x1f525;专栏&#xff1a;c笔记仓 朋友们大家好&#xff0c;本篇文章我们来到STL新的内容&#xff0c;stack和queue 目录 1. stack的介绍与使用函数介绍例题一&#xff1a;最小栈例题二&#xff1a;栈的压入、弹出队列栈的模…

springcloud按版本发布微服务达到不停机更新的效果

本文基于以下环境完成 spring-boot 2.3.2.RELEASEspring-cloud Hoxton.SR9spring-cloud-alibaba 2.2.6.RELEASEspring-cloud-starter-gateway 2.2.6.RELEASEspring-cloud-starter-loadbalancer 2.2.6.RELEASEnacos 2.0.3 一、思路 实现思路&#xff1a; 前端项目在请求后端接…

VSCode搭建内核源码阅读开发环境

0. 参考链接 使用VSCode进行linux内核代码阅读和开发_vscode阅读linux内核-CSDN博客 Ubuntu下的内核编译&#xff08;打造最精确的编译步骤&#xff09;_ubuntu 内核编译-CSDN博客 【Linux】&#xff08;Ubuntu&#xff09;内核编译 && 镜像制作_ubuntu 内核编译-CS…

Ubuntu部署jmeter与ant

为了整合接口自动化的持续集成工具&#xff0c;我将jmeter与ant都部署在了Jenkins容器中&#xff0c;并配置了build.xml 一、ubuntu部署jdk 1&#xff1a;先下载jdk-8u74-linux-x64.tar.gz&#xff0c;上传到服务器&#xff0c;这里上传文件用到了ubuntu 下的 lrzsz。 ubunt…