本文涉及知识点
C++BFS算法
LeetCode959. 由斜杠划分区域
在由 1 x 1 方格组成的 n x n 网格 grid 中,每个 1 x 1 方块由 ‘/’、‘’ 或空格构成。这些字符会将方块划分为一些共边的区域。
给定网格 grid 表示为一个字符串数组,返回 区域的数量 。
请注意,反斜杠字符是转义的,因此 ‘’ 用 ‘\’ 表示。
示例 1:
输入:grid = [" /“,”/ "]
输出:2
示例 2:
输入:grid = [" /“,” "]
输出:1
示例 3:
输入:grid = [“/\”,“\/”]
输出:5
解释:回想一下,因为 \ 字符是转义的,所以 “/\” 表示 /\,而 “\/” 表示 /。
提示:
n == grid.length == grid[i].length
1 <= n <= 30
grid[i][j] 是 ‘/’、‘’、或 ’ ’
题解
任意单格内部分四块,上下左右单格内编号分别为0,1,2,3。
如果此单格是空格,则0,1,2,3相连。
如果此空格是斜杠,则0,2 相连,1,3向量。
如果是反斜杠,则0,3相连,1,2相连。
各的编号:4*(r*n+c)+单格内编号。
可以直接用并集查找(并查集)计算连通区域,本文用BFS。
C++BFS
编号从小到大枚举各块,如果不属于任意区域,则将此块及相连的区域划分到cnt区域,并cnt++。
region[i]记录第i块,在那个区域,默认值-1,表示不属于任何区域。
neiBo是各块的邻接表。
BFS的状态表示:leves[i]记录和root通过i条边相连的块。
BFS的后续状态:通过next枚举neiBo[cur]
BFS的初始值:leves[0]={root}
BFS的返回值:无。
BFS的重复处理:利用region出重。
必须判断上一行是否存在,不能只判断编号是否n合法:
n=2时,块10的左边不存在,如果不判断c>0,则块10和块7连通。
代码
核心代码
class Solution {
public:
int regionsBySlashes(vector<string>& grid) {
const int N = grid.size();
const int N1 = N * N * 4;
vector<vector<int>> vNeiBo(N1);
auto Add = [&vNeiBo,&N1](int n1, int n2) {
if ((n1 < 0) || (n1 >= N1)) { return; }
if ((n2 < 0) || (n2 >= N1)) { return; }
vNeiBo[n1].emplace_back(n2);
vNeiBo[n2].emplace_back(n1);
};
//单格内部连接
for (int r = 0; r < N; r++) {
for (int c = 0; c < N; c++) {
const int iBase = 4 * (N * r + c);
if ('\\' != grid[r][c]) {
Add(iBase, iBase + 2);
Add(iBase+1, iBase + 3);
}
if ('/' != grid[r][c]) {
Add(iBase, iBase + 3);
Add(iBase + 1, iBase + 2);
}
}
}
//单格间连接
for (int r = 0; r < N; r++) {
for (int c = 0; c < N; c++) {
#define Mask(r,c,n) ((N*(r)+(c))*4+(n))
if (r > 0)
{
Add(Mask(r, c, 0), Mask(r - 1, c, 1));//本行上 上一行下
}
if (r + 1 < N)
{
Add(Mask(r, c, 1), Mask(r + 1, c, 0));//本行下 下一行上
}
if (c > 0)
{
Add(Mask(r, c, 2), Mask(r, c - 1, 3));//本行左 左列右
}
if (c + 1 < N)
{
Add(Mask(r, c, 3), Mask(r, c + 1, 2));//本行右 右列左
}
}
}
vector<int> region(N1, -1);
int cnt = 0;
for (int i = 0; i < N1; i++) {
if (-1 != region[i]) { continue; }
queue<int> que;
que.emplace(i);
region[i] = cnt;
while (que.size()) {
auto cur = que.front();
que.pop();
for (const auto& next : vNeiBo[cur]) {
if (-1 != region[next]) { continue; }
region[next] = cnt;
que.emplace(next);
}
}
cnt++;
}
return cnt;
}
};
单元测试
vector<string> grid;
TEST_METHOD(TestMethod1)
{
grid = { " " };
auto res = Solution().regionsBySlashes(grid);
AssertEx(1, res);
}
TEST_METHOD(TestMethod2)
{
grid = { "/" };
auto res = Solution().regionsBySlashes(grid);
AssertEx(2, res);
}
TEST_METHOD(TestMethod3)
{
grid = { "\\" };
auto res = Solution().regionsBySlashes(grid);
AssertEx(2, res);
}
TEST_METHOD(TestMethod11)
{
grid = { " /","/ " };
auto res = Solution().regionsBySlashes(grid);
AssertEx(2, res);
}
TEST_METHOD(TestMethod12)
{
grid = { " /"," " };
auto res = Solution().regionsBySlashes(grid);
AssertEx(1, res);
}
TEST_METHOD(TestMethod13)
{
grid = { "/\\","\\/" };
auto res = Solution().regionsBySlashes(grid);
AssertEx(5, res);
}
扩展阅读
我想对大家说的话 |
---|
工作中遇到的问题,可以按类别查阅鄙人的算法文章,请点击《算法与数据汇总》。 |
学习算法:按章节学习《喜缺全书算法册》,大量的题目和测试用例,打包下载。重视操作 |
有效学习:明确的目标 及时的反馈 拉伸区(难度合适) 专注 |
闻缺陷则喜(喜缺)是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。 |
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。 |
如果程序是一条龙,那算法就是他的是睛 |
失败+反思=成功 成功+反思=成功 |
视频课程
先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771
如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176
测试环境
操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。