bfs 和 dfs的相关的题目
题目: 给定一个数字列表,返回其所有可能的排列。
// premute(ans, nums, 0)
void permute(vector<vector<int> > &ans, vector<int> &nums, int k){
if(k==nums.size()-1){
ans.push_back(nums);
}
// 以k开头的所有排列
for(int i=k;i<nums.size();i++){
// 以每一个都作为开头,进行遍历
swap(nums[i],nums[k]);
permute(ans,nums,k+1);
// 回溯
swap(nums[i],nums[k]);
}
}
题目: 给定一个可能具有重复数字的列表,返回其所有可能的子集。
// 调用函数dfs(res, sub, , nums, 0)之前, nums 必须首先排序,
// sort(nums.begin(), nums.end());
void dfs(vector<vector<int>> &res, vector<int> &sub, vector<int> &nums, int k) {
res.push_back(sub);
for(int i= k; i < nums.size(); i++) {
// 跳过相同元素,
if(i != k && nums[i] == nums[i - 1]) continue;
sub.push_back(nums[i]);
dfs(res, sub, nums, i + 1);
// 回溯其他可能组合
sub.pop_back();
}
}
题目: 给一字串s和单词的字典dict,在字串中增加空格来构建一个句子,并且所有单词都来自字典。返回所有有可能的句子。
分析: 利用f[i]记录以i为起点的每个片段的终点j,并且片段要在字典中,然后从0位置开始搜索,每次给当前片段加上空格,然后以当前片段的末尾作为下一次搜索的头部,避免不必要的搜索。
vetor<int> f[1000];
vector<string> wordBreak(string &s, unordered_set<string> &wordDict) {
n = s.length();
int i, j;
// 遍历所有可能的(i,j)组合,是否在字典中
for (i = n - 1; i >= 0; --i) {
for (j = i + 1; j <= n; ++j) {
if (wordDict.find(s.substr(i, j - i)) != wordDict.end()) {
// 大家请思考不加这个条件和加条件有什么区别,
// if (j == n || f[j].size() > 0)
// f[i].push_back(j);
f[i].push_back(j);
}
}
}
dfs(0, s, "");
return res;
}
void dfs(int p, string s, string &now, vector<int> &res) {
if(p == s.size()) {
res.push_back(now);
return;
}
if(p > 0) { // 找到一个单词划分
now += " ";
}
// 遍历所有以p开头, 以j结尾的划分进行dfs
for(int i = 0; i < f[p].size(); i ++) {
dfs(f[p][i], s, now+s.substr(p, f[p][i]-p), res);
}
}
题目: 如果可以通过将 A 中的两个小写字母精确地交换位置 K 次得到与 B 相等的字符串,我们称字符串 A 和 B 的相似度为 K(K 为非负整数)。 给定两个字母异位词 A 和 B ,返回 A 和 B 的相似度 K 的最小值。 解析: 这是一个bfs的问题, 每次改变A的一个字符, 和B进行比较, 将改变后的A加入到候选队列中,直到所有出现A==B位置,得到此时的次数.
struct Node {
string s;
int step;
Node(string _s, int _step):s(_s),step(_step);
Node(){}
};
int kSimilarity(string &A, string &B) {
Node start(A, 0);
queue<Node> q;
set<string> vis;
q.push(start);
int ans = 0;
while(q.size()) {
Node str = q.front();
q.pop();
if(str.s == B) {
ans = str.step;
break;
}
int i = 0;
while(str[i] == B[i]) i ++;
for(int j = i + 1; j < B.size(); j ++) {
if(str[j] != B[j] && str[j] == B[i]) {
string temp = str;
swap(temp[i], temp[j]);
if(vis.find(temp) == vis.end()) {
q.push(Node(temp, str.step+1));
vis.insert(temp);
}
}
}
}
return ans;
}
题目: 给一个布尔类型的二维数组, 0 表示海, 1 表示岛。如果两个1是相邻的,那么我们认为他们是同一个岛.我们只考虑 上下左右 相邻. 求出岛屿的个数.
解析: 这就是无向图的联通块问题, 我们遍历所有是1的位置进行dfs(i,j), 并且将所有访问过的位置记录下来,如果当前位置是1,而且没有访问,则次数就加1.
void dfs(vector<vector<int>> &grid, int i, int j) {
if(i < 0 || i >= grid.size()) return;
if(j < 0 || j >= grid[0].size()) return;
if(!grid[i][j]) return;
grid[i][j] = 0;
// 四个方向搜索
dfs(grid, i-1, j);
dfs(grid, i+1, j);
dfs(grid, i, j-1);
dfs(grid, i, j+1);
}
int numIslands(vector<vector<int>> &grid) {
if (grid.empty() || grid[0].empty()) return 0;
int N = grid.size(), M = grid[0].size();
int cnt = 0;
for (int i = 0; i < N; ++i) {
for (int j = 0; j < M; ++j) {
if (grid[i][j]) {
dfs(grid, i, j);
++cnt;
}
}
}
return cnt;
}
题目: 给定n个不同的正整数,整数k(1<= k <= n)以及一个目标数字。 在这n个数里面找出K个数,使得这K个数的和等于目标数字,你需要找出所有满足要求的方案。 解析: dfs(i, k, target) 每次怕判断是够使用第i个值, 如果使用, dfs(i+1,k-1,target-arr[i]) 如果不使用, dfs(i+1,k,target), if target == 0, 则表示满足要求,存储结果
void dfs(vector<int> A, int i, int k, int target, vector<int> &now, vector<vector<int>> &res) {
if(i > A.size() || target < 0 || k < 0) return;
if(target == 0 && k==0) {
res.push_back(now);
return;
}
// user
now.push_back(A[i]);
dfs(A, i+1, k-1, target-A[i], now, res);
now.pop_back();
// not use i
dfs(A, i+1, k, target, now, res);
}
vector<vector<int>> kSumII(vector<int> &A, int k, int target) {
// write your code here
vector<vector<int>> res;
vector<int> now;
dfs(A,0,k,target,now,res);
return res;
}
题目: 给出两个单词(start和end)和一个字典,找出从start到end的最短转换序列,输出最短序列的长度。
变换规则如下:
- 每次只能改变一个字母。
- 变换过程中的中间单词必须在字典中出现。(起始单词和结束单词不需要出现在字典中)
解析: 使用bfs进行变换,每一修改一个字符['a'--'z'],判断是否在字典中,并记录当前的步数.
int ladderLength(string &start, string &end, unordered_set<string> &dict) {
int length = 2;
if(start == end) return 1;
queue<string> q;
q.push(start);
while(!q.empty()){
int size = q.size();// 对每一个层以此处理, 这一个的都步数一样
for(int i=0;i<size;i++){
string tmp = q.front();
q.pop();
// 遍历tmp的所有的字符,进行26个字符的变换
for(int j=0;j<tmp.size();j++){
// 要记录老字符,因为最后要恢复
char oldc = tmp[j];
for(char c='a';c<='z';c++){
if(tmp[j] == c) continue;
tmp[j] = c;
//验证是否已经满足条件
if(tmp == end) return length;
// 变换的单词是否在字典中
if(dict.find(tmp) != dict.end()){
q.push(tmp);
dict.erase(tmp); // 防止多次使用
}
}
// 恢复当前的变化,这个不变,变化下一个,
tmp[j] = oldc;
}
}
length ++;
}
return length;
}
题目: 给出一个二维的字母板和一个单词,寻找字母板网格中是否存在这个单词。单词可以由按顺序的相邻单元的字母组成,其中相邻单元指的是水平或者垂直方向相邻。每个单元中的字母最多只能使用一次。
样例
样例 1:
输入:["ABCE","SFCS","ADEE"],"ABCCED"
输出:true
解释:
[
A B C E
S F C S
A D E E
]
(0,0)A->(0,1)B->(0,2)C->(1,2)C->(2,2)E->(2,1)D
样例 2:
输入:["z"],"z"
输出:true
解释:
[ z ]
(0,0)z
bool dfs(int i, int j, int k, vector<vector<char>> &board, string word, vector<vector<int>> &vis) {
if(board[i][j] == word[k]) {
++ k;
if(k == word.size()) {
return true;
}
}
else return false;
bool flag = false;
vis[i][j] = 1;
if(i-1 >=0 && (!vis[i-1][j]) && board[i-1][j] == word[k]) flag = flag | dfs(i-1, j, k, board, word, vis);
if(flag) return flag;
if(i+1 < board.size() && (!vis[i+1][j]) && board[i+1][j] == word[k]) flag = flag | dfs(i+1, j, k, board, word, vis);
if(flag) return flag;
if(j-1 >= 0 && (!vis[i][j-1]) && board[i][j-1] == word[k]) flag = flag | dfs(i, j-1, k, board, word, vis);
if(flag) return flag;
if(j+1 <= board[0].size() && (!vis[i][j+1]) && board[i][j+1] == word[k]) flag = flag | dfs(i, j+1, k, board, word, vis);
// 下次使用标记
vis[i][j] = 0;
return flag;
}
bool exist(vector<vector<char>> &board, string &word) {
if(board.empty() || board[0].size() == 0) return false;
bool res = false;
vector<vector<int>> vis(board.size(), vector<int>(board[0].size(), 0));
for(int i = 0; i < board.size(); i ++) {
for(int j = 0; j < board[i].size(); j ++) {
if(word[0] == board[i][j] && dfs(i,j,0,board,word, vis)){
return true;
}
}
}
return res;
}
题目: 给一个字符串,你可以选择在一个字符或两个相邻字符之后拆分字符串,使字符串由仅一个字符或两个字符组成,输出所有可能的结果.
解析: dfs(s) = dfs(s-1) + dfs(s-2)
void dfs(int i, string s, vector<string> &now, vector<vector<string>> &res) {
if(i == s.size()) {
res.push_back(now);
return;
}
if(s.size() - i == 1) {
now.push_back(s.substr(i, 1));
dfs(i+1,s,now,res);
now.pop_back();
return;
}
if(s.size() - i >= 2) {
now.push_back(s.substr(i, 1));
dfs(i+1,s,now,res);
now.pop_back();
now.push_back(s.substr(i, 2));
dfs(i+2,s,now,res);
now.pop_back();
}
}
vector<vector<string>> splitString(string& s) {
vector<string> now;
vector<vector<string>> res;
dfs(0,s,now,res);
return res;
}
题目: 给定一个字符串S,将S切分成每一个子串都是回文串,返回所有可能的结果.
Input : s = "bcc"
Output : [["b", "c", "c"], ["b", "cc"]]
bool checkPalindrome(string str) {
int len = str.length();
len--;
for (int i=0; i<len; i++) {
if (str[i] != str[len]) return false;
len--;
}
return true;
}
void addStrings(vector<vector<string> > &v, string &s,
vector<string> &temp, int index) {
int len = s.length();
string str;
vector<string> current = temp;
if (index == 0) temp.clear();
for (int i = index; i < len; ++i) {
str = str + s[i];
if (checkPalindrome(str)) {
temp.push_back(str);
if (i+1 < len)
addStrings(v,s,temp,i+1);
else
v.push_back(temp);
temp = current;
}
}
return;
}