zoukankan      html  css  js  c++  java
  • Acwing-----算法提高课第二章搜索(五)

    1. 优化搜索顺序

      大部分情况下,我们应该优先搜索分支较少的节点;

    2. 排除等效冗余

    3. 可行性剪枝

    4. 最优性剪枝

    5. 记忆化搜索(DP)

    165. 小猫爬山

    #include <iostream>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    const int N = 20;
    
    int n, m, w[N], sum[N], ans = N;
    
    void dfs(int u, int k) {
        // 最优性剪枝
        if (k >= ans) return;
        if (u == n) {
            ans = k;
            return ;
        }
        for (int i = 0; i < k; ++i) {
            if (sum[i] + w[u] <= m) { // 可行性剪枝
                sum[i] += w[u];
                dfs(u + 1, k);
                sum[i] -= w[u];
            }
        }
        sum[k] = w[u];
        dfs(u + 1, k + 1);
        sum[k] = 0;
    }
    
    int main() {
        cin >> n >> m;
        for (int i = 0; i < n; ++i) cin >> w[i];
        
        //优化搜索顺序
        sort(w, w + n);
        reverse(w, w + n);
        
        dfs(0, 0);
        cout << ans << endl;
        return 0;
    }
    

    166. 数独

    #include <cstring>
    #include <algorithm>
    #include <iostream>
    using namespace std;
    const int N = 9, M = 1 << N;
    
    int ones[M], map[M];
    int row[N], col[N], cell[3][3];
    char str[100];
    
    void init() {
        for (int i = 0; i < N; ++i) row[i] = col[i] = (1 << N) - 1;
        for (int i = 0; i < 3; ++i) {
            for (int j = 0; j < 3; ++j) {
                cell[i][j] = (1 << N) - 1;
            }
        }
    }
    
    void draw(int x, int y, int t, bool is_set) {
        if (is_set) str[x * N + y] = '1' + t;
        else str[x * N + y] = '.';
        
        int v = 1 << t;
        if (!is_set) v = -v;
        
        row[x] -= v;
        col[y] -= v;
        cell[x / 3][y / 3] -= v;
    }
    
    int lowbit(int x) {
        return x & -x;
    }
    
    int get(int x, int y) {
        return row[x] & col[y] & cell[x / 3][y / 3];
    }
    
    bool dfs(int cnt) {
        if (!cnt) return true;
        int minv = 10;
        int x, y;
        for (int i = 0; i < N; ++i) {
            for (int j = 0; j < N; ++j) {
                if (str[i * N + j] == '.') {
                    int state = get(i, j);
                    if (ones[state] < minv) {
                        minv = ones[state];
                        x = i, y = j;
                    }
                }
            }
        }
        int state = get(x, y);
        for (int i = state; i; i -= lowbit(i)) {
            int t = map[lowbit(i)];
            draw(x, y, t, true);
            if (dfs(cnt - 1)) return true;
            draw(x, y, t, false);
        }
        return false;
    }
    
    int main() {
        for (int i = 0; i < N; ++i) map[1 << i] = i;
        for (int i = 0; i < 1 << N; ++i) {
            for (int j = 0; j < N; ++j) {
                ones[i] += i >> j & 1;
            }
        } 
        while (cin >> str, str[0] != 'e') {
            init();
            
            int cnt = 0;
            for (int i = 0, k = 0; i < N; ++i) {
                for (int j = 0; j < N; ++j, ++k) {
                    if (str[k] != '.') {
                        int t = str[k] - '1';
                        draw(i, j, t, true);
                    } else cnt++;
                }
            }
            dfs(cnt);
            cout << str << endl;
        }
        return 0;
    }
    

    167. 木棒

    #include <iostream>
    #include <algorithm>
    #include <cstring>
    using namespace std;
    
    const int N = 70;
    
    int n;
    int w[N], sum, length;
    bool st[N];
    
    bool dfs(int u, int s, int start) {
        if (u * length == sum) return true;
        if (s == length) return dfs(u + 1, 0, 0);
        
        // 剪枝3-1,从start开始枚举
        for (int i = start; i < n; ++i) {
            if (st[i]) continue;
            if (s + w[i] > length) continue; // 可行性剪枝
            
            st[i] = true;
            if (dfs(u, s + w[i], i + 1)) return true;
            st[i] =  false;
            
            // 剪枝3-3
            if (!s) return false;
            
            // 剪枝3-4
            if (s + w[i] == length) return false;
            
            // 剪枝3-2
            int j = i;
            while (j < n && w[j] == w[i]) j++;
            i = j - 1;
        }
        return false;
    }
    
    int main() {
        while (cin >> n, n) {
            memset(st, 0, sizeof st);
            sum = 0;
            for (int i = 0; i < n; ++i)  {
                cin >> w[i];
                sum += w[i];
            }
            // 剪枝2,优化搜索顺序
            sort(w, w + n);
            reverse(w, w + n);
            
            length = 1;
            while (true) {
                // 剪枝1
                if (sum % length == 0 && dfs(0, 0, 0)) {
                    cout << length << endl;
                    break;
                }
                length++;
                if (length > sum) break;
            }
        }
        return 0;
    }
    

    168. 生日蛋糕

    #include <iostream>
    #include <algorithm>
    #include <cstring>
    #include <cmath>
    using namespace std;
    
    const int N = 25, INF = 1e9;
    
    int n, m;
    int minv[N], mins[N];
    int R[N], H[N];
    int ans = INF;
    
    void dfs(int u, int v, int s) {
        if (v + minv[u] > n) return;
        if (s + mins[u] >= ans) return;
        if (s + 2 * (n - v) / R[u + 1] >= ans) return;
        
        if (!u) {
            if (v == n) ans = s;
            return;
        }
        
        for (int r = min(R[u + 1] - 1, (int)sqrt(n - v)); r >= u; r--) {
            for (int h = min(H[u + 1] - 1, (n -v) / r / r); h >= u; h--) {
                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);
            }
        }
    }
    
    int main() {
        cin >> n >> m;
        for (int i = 1; i <= m; ++i) {
            minv[i] = minv[i - 1] + i * i * i;
            mins[i] = mins[i - 1] + 2 * i * i; 
        }
        R[m + 1] = H[m + 1] = INF;
        dfs(m, 0, 0);
        if (ans == INF) ans = 0;
        cout << ans << endl;
        
        return 0;
    }
    
  • 相关阅读:
    Javascript创建对象的几种方式【转】
    log4net轻松使用日期作为动态文件名【转】
    使用jquery的lazy loader插件实现图片的延迟加载
    Oracle通用分页存储过程的创建与使用
    Oracle 11g R2的卸载与重装
    Oracle的rownum的原理和使用【转】
    Remoting客户端和服务端两种方式的调用总结
    动态执行SQL语句
    Trie模板
    Dijkstrapriority_queue
  • 原文地址:https://www.cnblogs.com/clown9804/p/13765500.html
Copyright © 2011-2022 走看看