算法提高-搜索-DFS之剪枝与优化

DFS之剪枝与优化

  • DFS之剪枝与优化
    • AcWing 165. 小猫爬山
    • AcWing 166. 数独
    • AcWing 167. 木棒
    • AcWing 168. 生日蛋糕

DFS之剪枝与优化

AcWing 165. 小猫爬山

DFS的五种剪枝方法
(1)优化搜索顺序
(2)排除等效冗余
(3)可行性剪枝
(4)最优性剪枝
(5)记忆化搜索(这个放在dp里面说了)
下面这题涉及三个剪枝

#include 
#include 
#include 
using namespace std;

const int N = 20;

int n;
int maxW;
int sum[N];
int w[N];
int ans = N;//这里debug很久,题目找的是最小值,初始化的时候要给他定义一个最大值

void dfs(int u, int k)
{
    // (3)最优性剪枝
    if (k >= ans) return ;
    if (u == n)
    {
        ans = k;
        return ;
    }
    //1.判断能否加入现在已经存在的缆车
    //遍历缆车
    for (int i = 0; i < k; i ++ )
    {
        // (2)可行性剪枝
        if (sum[i] + w[u] <= maxW)
        {
            sum[i] += w[u];
            dfs(u + 1, k);
            sum[i] -= w[u];//回溯
        }
    }
    //2.判断是否需要新开一个缆车
    sum[k] = w[u];//因为我们缆车的下标是从0开始的
    dfs(u + 1, k + 1);
    //sum[k] = 0;   sum是全局变量,但这不是sum += w[u],其实不用回溯,我们返回到k - 1层的时候会直接重新赋值覆盖sum[k]
}

int main()
{
    cin >> n >> maxW;
    for (int i = 0; i < n; i ++ ) cin >> w[i];
    // (1)优化搜索顺序
    sort(w, w + n);
    reverse(w, w + n);
    dfs(0, 0);
    cout << ans;
    return 0;
}

AcWing 166. 数独

AcWing 167. 木棒

6种剪枝方法:1, 2, 3-1, 3-2, 3-3, 3-4
算法提高-搜索-DFS之剪枝与优化_第1张图片

#include 
#include 
#include 

using namespace std;
const int N = 64 + 5;

int sum;//所有小棒加起来的总长度,也是所有大棒加起来的总长度
int length;//枚举每一根大棒的长度
int n;
int stick[N];//记录每根木棍的长度
int st[N];


bool dfs(int u, int cur, int start) 
{
    if (u * length == sum) return true;//找到一组满足题意的解
    if (cur == length) return dfs(u + 1, 0, 0);//第u根木棒组合完毕,枚举第u+1根,直到u * length == sum才代表满足题意

    for (int i = start; i < n; i ++ )
    {
        if (st[i]) continue;
        if (stick[i] + cur > length) continue;//可行性剪枝
        
        st[i] = true;
        if (dfs(u, cur + stick[i], i + 1)) return true;//这一步相当编译器帮我们回溯
        st[i] = false;//手动回溯
        
        //剪枝 3-3 和 3-4,
        //走到了这一步说明之前dfs return false了,我们判断一下是将小棍放在大棍的哪个位置失败了
        //根据3-3和3-4,如果是放在大棍的第一个位置和放在大棍的最后一个位置失败了,那么就直接不同继续判断下去了
        //当前这个小棍stick[i]也一定不能放在后续的其他大棍中,必找不到一组合法解
        if (cur == 0 || cur + stick[i] == length) return false;
        
        //剪枝3-2如果当前的小棍失败了,那么略过枚举stick[]中后面和它长度一样的小棍
        int j = i + 1;
        while (j < n && stick[j] == stick[i]) j ++ ;
        i = j - 1;//上面的while能退出说明stick[i]已经!=stick[j]了,所有要 -1
    }
    return false;
}



int main ()
{
    while (cin >> n, n != 0)
    {
        //多组数据,sum和st要置0
        memset(st, 0, sizeof st);
        sum = 0;
        for (int i = 0; i < n; i ++ )
        {
            cin >> stick[i];
            sum += stick[i];
        }
        
        sort(stick, stick + n);//优化搜索顺序,先枚举长的木棒,这样后面需要搜索的分支可以减少些
        reverse(stick, stick + n);
        
        length = 1;
        while (true)
        {
            if (sum % length == 0 && dfs(0, 0, 0))
            {
                cout << length << endl; //题目求的是原始木棒的最小长度,因此我们length从1开始枚举即可,只要找到一组合法的解就break
                break;
            }
            length ++ ;//走到这一步了说明之前没break没找到一组合法的解,那么继续枚举
        }
    }
    return 0;
}

AcWing 168. 生日蛋糕

这题挺难的,我反正糊着写完了建议配合y总讲解和这篇题解复习
算法提高-搜索-DFS之剪枝与优化_第2张图片

#include 
#include 
#include 
#include 

using namespace std;

const int M = 25, INF = 1e9;
int n, m;
int R[M], H[M];//记录每一层的半径和高度
int minv[M], mins[M];//记录前i层的最小体积和,最小面积和
int ans = INF;

void dfs(int u, int v, int s)
{
    //可行性剪枝
    if (v + minv[u] > n) return ;
    if (s + mins[u] >= ans) return ;
    //这个是用公式推出来的1-u层蛋糕的面积的最小值
    if (s + 2 * (n - v) / R[u + 1] >= ans) return ;
    
    if (u == 0)//倒着枚举的,所以在第0层收集结果
    {
        if (v == n) ans = s;//前面没return,说明这里的s一定小于ans,直接赋值就行了,我们这里的ans一定是最小的
        return ;//这个return不能放在if里面,因为也许u == 0的时候v!=n,return要是放在里面如果无解就会TLE
    }
    
    //优化搜索顺序,r是二次方的,先搜r再搜h,分支会少
    for (int r = min(R[u + 1] - 1, (int)sqrt(n - v)); r >= u; r -- )//min函数一定要都是int,所以要强转
        for (int h = min(H[u + 1] -1, n - v / r / r); h >= u; h -- )//倒着枚举的要--
        {
            //特判一下,如果是m层我们直接加上整个上表面积就行,后面再枚举每个侧表面积
            int t = 0;
            if (u == m) t = r * r;
            R[u] = r, H[u] = h;
            dfs(u - 1, v + r * r * h, s + 2 * r * h + t);//2 * r * h是这一层的侧面积,t是特判出来的上表面积,就是一整个圆盘,一组解只要加1次就行
        }
}


int main ()
{
    cin >> n >> m;
    
    //预处理
    for (int i = 1; i <= m; i ++ )
    {
        minv[i] = minv[i] + i * i * i;//r * r * h
        mins[i] = mins[i] + 2 * i * i;//2 * r * r ,题目说了方便计算pi省略
    }
    //m+1层不存在,作为哨兵,减少边界情况的判断
    R[m + 1] = H[m + 1] = INF;//不加这个会报Float Point Exception,但这个y总说是除0了
    
    //优化搜索顺序,先从最大的一层搜,分支会少
    dfs(m, 0, 0);
    
    if (ans == INF) ans = 0;//特判无解的情况
    cout << ans;
    return 0;
}

你可能感兴趣的:(深度优先,算法,剪枝,c++,蓝桥杯)