2022.02.27刺杀大使
题目描述
某组织正在策划一起对某大使的刺杀行动。他们来到了使馆,准备完成此次刺杀,要进入使馆首先必须通过使馆前的防御迷阵。
迷阵由 n×m 个相同的小房间组成,每个房间与相邻四个房间之间有门可通行。在第 n 行的 m 个房间里有 m 个机关,这些机关必须全部打开才可以进入大使馆。而第 1 行的 m 个房间有 m 扇向外打开的门,是迷阵的入口。除了第 1 行和第 n 行的房间外,每个房间都被使馆的安保人员安装了激光杀伤装置,将会对进入房间的人造成一定的伤害。第 i 行第 j 列 造成的伤害值为$ p_{i,j}$ (第 1 行和第 n 行的 p 值全部为 0)。
现在某组织打算以最小伤害代价进入迷阵,打开全部机关,显然,他们可以选 择任意多的人从任意的门进入,但必须到达第 n 行的每个房间。一个士兵受到的伤害值为他到达某个机关的路径上所有房间的伤害值中的最大值,整个部队受到的伤害值为所有士兵的伤害值中的最大值。现在,这个恐怖组织掌握了迷阵的情况,他们需要提前知道怎么安排士兵的行进路线可以使得整个部队的伤害值最小。
输入格式
第一行有两个整数 n,m,表示迷阵的大小。
接下来 n 行,每行 m 个数,第 i 行第 j 列的数表示 pi,jp_{i,j}pi,j
输出格式
输出一个数,表示最小伤害代价。
样例输入
4 2
0 0
3 5
2 4
0 0
样例输出
3
思路
题目中即求从上到下的一条路径,满足路径上的最大数字最小。由于答案已经限定在了矩阵中的所有数字,所以可以采用二分查找解该题。
首先使用二分查找遍历所有可能的解,再使用dfs遍历该数是否可行。
代码
int m, n;
int a[][];
int[][] dir = {
{0,1},
{0,-1},
{1,0},
{-1,0},
};
boolean[][] vis;
int[] t;
int tar;
boolean check = false;
boolean isIn(int i, int j) {
return i>=0&&i<m&&j>=0&&j<n;
}
void dfs(int i, int j) {
if(check) return;
if(i == m-1) {
check = true;
} else {
for(int k = 0; k < dir.length; k++) {
int nx = i+dir[k][0];
int ny = j+dir[k][1];
if(isIn(nx,ny) && !vis[nx][ny] && a[nx][ny] <= tar) {
vis[nx][ny] = true;
dfs(nx,ny);
}
}
}
}
boolean checkFun() {
for(int i = 0; i < m; i++) for(int j = 0; j < n; j++) vis[i][j] = false;
for(int i = 0; i < n; i++) {
check = false;
if(!vis[1][i] && a[1][i] <= tar) {
vis[1][i] = true;
dfs(1,i);
if(check) return true;
}
}
return false;
}