zoukankan      html  css  js  c++  java
  • 【LOJ】 #2009. 「SCOI2015」小凸玩密室

    题解

    神仙dp啊QAQ

    我们发现我们需要枚举一个起点,遍历完它所有的儿子然后向上爬

    (f[i][j])表示第i个点的子树全部处理完之后到达i深度为j的祖先的兄弟处

    我们只需要对叶子节点和只有一个儿子的点特殊讨论,因为所有的向上爬都是从叶子爬的

    转移的时候只要枚举从两个儿子里哪个爬上取就好了

    (g[i][j])表示第i个点的子树全部处理完之后到达i深度为j的祖先处

    转移个f数组类似,但是要用到f数组

    枚举每个点当起点的时候,遵循x的父亲->x的兄弟->x的父亲的父亲->x的父亲的兄弟这样顺序来覆盖即可

    代码

    #include <bits/stdc++.h>
    //#define ivorysi
    #define enter putchar('
    ')
    #define space putchar(' ')
    #define fi first
    #define se second
    #define pb push_back
    #define mp make_pair
    #define eps 1e-8
    #define mo 974711
    #define MAXN 200005
    #define pii pair<int,int>
    using namespace std;
    typedef long long int64;
    typedef double db;
    template<class T>
    void read(T &res) {
        res = 0;char c = getchar();T f = 1;
        while(c < '0' || c > '9') {
    	if(c == '-') f = -1;
    	c = getchar();
        }
        while(c >= '0' && c <= '9') {
    	res = res * 10 + c - '0';
    	c = getchar();
        }
        res *= f;
    }
    template<class T>
    void out(T x) {
        if(x < 0) {putchar('-');x = -x;}
        if(x >= 10) {
    	out(x / 10);
        }
        putchar('0' + x % 10);
    }
    int N,dep[MAXN];
    int64 a[MAXN],b[MAXN],dis[MAXN],f[MAXN][18],g[MAXN][18];
    void Solve() {
        read(N);
        for(int i = 1 ; i <= N ; ++i) read(a[i]);
        for(int i = 2 ; i <= N ; ++i) read(b[i]);
        for(int i = 1 ; i <= N ; ++i) {
    	dep[i] = dep[i >> 1] + 1;
    	dis[i] = dis[i >> 1] + b[i];
        }
        for(int u = N ; u >= 1 ; --u) {
    	if((u << 1) > N) {
    	    for(int j = dep[u]; j >= 2 ; --j) {
    		int fa = u >> (dep[u] - j + 1),x = (u >> (dep[u] - j)) ^ 1;
    		f[u][j] = (dis[u] + dis[x] - 2 * dis[fa]) * a[x];
    	    }
    	}
    	else if((u << 1) == N) {
    	    for(int j = dep[u] ; j >= 2 ; --j) {
    		f[u][j] = f[u << 1][j] + a[u << 1] * b[u << 1];
    	    }
    	}
    	else {
    	    int lc = u << 1,rc = u << 1 | 1;
    	    for(int j = dep[u] ; j >= 2 ; --j) {
    		f[u][j] = min(a[lc] * b[lc] + f[lc][dep[lc]] + f[rc][j],a[rc] * b[rc] + f[rc][dep[rc]] + f[lc][j]);
    	    }
    	}
        }
        for(int u = N ; u >= 1 ; --u) {
    	if((u << 1) > N) {
    	    for(int j = dep[u] - 1; j >= 0 ; --j) {
    		int fa = u >> (dep[u] - j);
    		g[u][j] = (dis[u] - dis[fa]) * a[fa];
    	    }
    	}
    	else if((u << 1) == N) {
    	    for(int j = dep[u] - 1 ; j >= 0 ; --j) {
    		g[u][j] = g[u << 1][j] + a[u << 1] * b[u << 1];
    	    }
    	}
    	else {
    	    int lc = u << 1,rc = u << 1 | 1;
    	    for(int j = dep[u] - 1 ; j >= 0 ; --j) {
    		g[u][j] = min(a[lc] * b[lc] + f[lc][dep[lc]] + g[rc][j],a[rc] * b[rc] + f[rc][dep[rc]] + g[lc][j]);
    	    }
    	}
        }
        int64 ans = 1e18;
        for(int u = N ; u >= 1 ; --u) {
    	int64 res = g[u][dep[u] - 1];
    	for(int x = u ; x > 1 ; x >>= 1) {
    	    int t = x ^ 1;
    	    if(t > N) res += a[x >> 2] * b[x >> 1];
    	    else res += a[t] * b[t] + g[t][dep[t] - 2];
    	}
    	ans = min(ans,res);
        }
        out(ans);enter;
    }
    int main() {
    #ifdef ivorysi
        freopen("f1.in","r",stdin);
    #endif
        Solve();
        return 0;
    }
    
  • 相关阅读:
    PyQt5 控件学习(一个一个学习之QCommandLinkButton)
    多任务--线程
    PyQt5 控件学习(一个一个学习之QPushButton)
    PyQt5 控件学习(一个一个学习之QAbstractButton)
    再测我心中的事
    花了两天时间,整理了代码,封装了逻辑
    我现在发现,我写代码有严重的问题
    2014年8月2日0时13分22秒
    2014年8月2日15时13分4秒
    交警与货车司机
  • 原文地址:https://www.cnblogs.com/ivorysi/p/9159473.html
Copyright © 2011-2022 走看看