文章目录
- 字母大小写全排列
- 优美的排列
- N 皇后
- 有效的数独
字母大小写全排列
题目:字母大小写全排列
思路
对每个位置的字符有两种情况
- 不修改:数字字符,直接递归下一层;
- 修改:字母字符,大写改小写、小写改大写后,递归下一层;
C++代码
class Solution
{
string path;
vector<string> ret;
public:
vector<string> letterCasePermutation(string s)
{
dfs(s, 0);
return ret;
}
void dfs(string& s, int pos)
{
if(pos == s.size())
{
ret.push_back(path);
return;
}
int ch = s[pos];
// 不改变
path.push_back(ch);
dfs(s, pos + 1);
path.pop_back();
// 改变
if(ch < '0' || ch > '9')
{
char t = change(ch);
path.push_back(t);
dfs(s, pos + 1);
path.pop_back();
}
}
char change(char ch)
{
if(ch >= 'a' && ch <= 'z') ch -= 32;
else ch += 32;
return ch;
}
};
优美的排列
题目:优美的排列
思路
我们可以使用回溯法解决本题,从左向右依次向目标排列中放入数即可
- 使用
check
数组来跟踪哪些数字已经被使用过。 - 在每次递归中,检查当前排列是否符合优美排列的条件。
- 基于回溯生成所有排列: 在排列中逐步填充数字,并检查每一步是否满足条件。
- 统计符合条件的排列数量。
C++代码
class Solution
{
bool check[16];
int ret;
public:
int countArrangement(int n)
{
// 从下表为 1 开始枚举
dfs(1, n);
return ret;
}
void dfs(int pos, int n)
{
if(pos == n + 1)
{
ret++;
return;
}
for(int i = 1; i <= n; i++)
{
if(!check[i] && (pos % i == 0 || i % pos == 0))
{
check[i] = true;
dfs(pos + 1, n);
check[i] = false;
}
}
}
};
N 皇后
题目:N 皇后
思路
在第一行放置第一个皇后,然后遍历棋盘的第二行,在合法的位置放置第二个皇后,再遍历第三行,以此类推,直到放置了n个皇后为止
- 从第一行开始,尝试在每一列放置皇后
- 对于每个放置位置,检查该位置的列和两个对角线是否已经被占用
- 如果该位置未被占用,则放置皇后,并标记相应的列和对角线为已占用
- 递归地尝试在下一行放置皇后
- 如果在当前行无法放置皇后(即所有列和对角线都被占用),则回溯,撤销上一行的皇后放置,并尝试在当前行的下一个位置放置皇后
- 当所有行都成功放置了皇后时,保存当前排列
C++代码
class Solution
{
bool checkCol[10], checkDig1[20], checkDig2[20];
vector<vector<string>> ret;
vector<string> path;
int n;
public:
vector<vector<string>> solveNQueens(int _n)
{
n = _n;
path.resize(n, string(n, '.'));
dfs(0);
return ret;
}
void dfs(int row)
{
if(row == n)
{
ret.push_back(path);
return;
}
for(int col = 0;col < n; col++)
{
if(!checkCol[col] && !checkDig1[row-col+n] && !checkDig2[row+col])
{
path[row][col] = 'Q';
checkCol[col] = checkDig1[row - col + n] = checkDig2[row + col] = true;
dfs(row + 1);
path[row][col] = '.';
checkCol[col] = checkDig1[row - col + n] = checkDig2[row + col] = false;
}
}
}
};
有效的数独
题目:有效的数独
思路
- 遍历棋盘的每一个格子
- 如果当前格子不是空字符
.
,则提取该字符代表的数字board[i][j] - '0'
- 检查这个数字在当前行、当前列和当前3x3子网格中是否已经出现过(通过查看
row[i][num]
、col[j][num]
和grid[i/3][j/3][num]
是否为0
) - 如果这个数字在任何一处已经出现过,则返回false,表示数独无效
- 如果这个数字在所有检查的地方都未出现过,则将其在
row
、col
和grid
中标记为已出现过(即将相应的位置设为1
) - 如果遍历完所有格子都没有发现重复数字,则返回
true
,表示数独有效
C++代码
class Solution
{
int row[9][10];
int col[9][10];
int grid[3][3][10];
public:
bool isValidSudoku(vector<vector<char>>& board)
{
for(int i = 0; i < 9; i++)
{
for(int j = 0; j < 9; j++)
{
if(board[i][j] != '.')
{
int num = board[i][j] - '0';
if(!row[i][num] && !col[j][num] && !grid[i/3][j/3][num])
{
row[i][num] = col[j][num] = grid[i/3][j/3][num]=1;
}
else
return false;
}
}
}
return true;
}
};