zoukankan      html  css  js  c++  java
  • 【BZOJ】4380: [POI2015]Myjnie

    题解

    区间dp,先离散化所有价值

    (f[i][j][k])表示([i,j])区间里最小值为(k)的价值最大是多少

    只考虑(i <= a <= b <= j)的区间,枚举中间点(h),然后如果这个区间包括h就统计在内
    (f[i][j][k] = max(f[i][h - 1][ >=k] + cost(k) + f[h + 1][j][>=k]))
    构造答案的时候通过记录每个点前一个转移点来递归即可

    代码

    #include <bits/stdc++.h>
    #define fi first
    #define se second
    #define pii pair<int,int>
    #define pdi pair<db,int>
    #define mp make_pair
    #define pb push_back
    #define enter putchar('
    ')
    #define space putchar(' ')
    #define eps 1e-8
    #define mo 974711
    #define MAXN 500005
    //#define ivorysi
    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) {x = -x;putchar('-');}
        if(x >= 10) {
    	out(x / 10);
        }
        putchar('0' + x % 10);
    }
    int N,M;
    int id[500005],val[4005],tot,ans[55];
    int a[4005],b[4005],c[4005],cnt[4005][55];
    int f[55][55][4005],g[55][55][4005],pre[55][55][4005];
    vector<int> p[55][55];
    void dfs(int l,int r,int v,int t = 1) {
        if(l > r) return;
        for(int k = t ; k <= tot ; ++k) {
    	if(f[l][r][k] == v) {
    	    if(l == r) {ans[l] = val[k];return;}
    	    int p = pre[l][r][k];
    	    if(!p) {
    		for(int i = l ; i <= r ; ++i) ans[i] = val[k];
    		return;
    	    }
    	    ans[p] = val[k];
    	    dfs(l,p - 1,g[l][p - 1][k],k);dfs(p + 1,r,g[p + 1][r][k],k);
    	    return;
    	}
        }
    }
    void Init() {
        read(N);read(M);
        for(int i = 1 ; i <= M ; ++i) {
    	read(a[i]);read(b[i]);read(c[i]);
    	val[i] = c[i];
    	for(int k = 1 ; k <= N ; ++k) {
    	    for(int h = k ; h <= N ; ++h) {
    		if(a[i] >= k && b[i] <= h) p[k][h].pb(i);
    	    }
    	}
        }
        sort(val + 1,val + M + 1);
        tot = unique(val + 1,val + M + 1) - val - 1;
        for(int i = 1 ; i <= tot ; ++i) {
    	id[val[i]] = i;
        }
    }
    void Solve() {
        for(int d = 1 ; d <= N ; ++d) {
    	for(int i = 1 ; i <= N ; ++i) {
    	    int j = i + d - 1;
    	    if(j > N) break;
    	    memset(cnt,0,sizeof(cnt));
    	    int s = p[i][j].size();
    	    for(int k = 0 ; k < s ; ++k) {
    		int t = p[i][j][k];
    		cnt[id[c[t]]][a[t]]++;cnt[id[c[t]]][b[t] + 1]--;
    	    }
    	    for(int k = tot ; k >= 1 ; --k) {
    		for(int h = i ; h <= j ; ++h) cnt[k][h] += cnt[k][h - 1];
    		for(int h = i ; h <= j ; ++h) cnt[k][h] += cnt[k + 1][h];
    	    }
    	    for(int k = tot ; k >= 1 ; --k) {
    		for(int h = i ; h <= j ; ++h) {
    		    if(f[i][j][k] < cnt[k][h] * val[k] + g[i][h - 1][k] + g[h + 1][j][k]) {
    			pre[i][j][k] = h;
    			f[i][j][k] = cnt[k][h] * val[k] + g[i][h - 1][k] + g[h + 1][j][k];
    		    }
    		}
    		g[i][j][k] = max(g[i][j][k + 1],f[i][j][k]);
    	    }
    	}
        }
        int r = 0;
        for(int i = 1 ; i <= tot ; ++i) r = max(r,f[1][N][i]);
        out(r);enter;
        dfs(1,N,r);
        for(int i = 1 ; i <= N ; ++i) {
    	out(ans[i]);
    	i == N ? enter : space;
        }
    }
    int main() {
    #ifdef ivorysi
        freopen("f1.in","r",stdin);
    #endif
        Init();
        Solve();
        return 0;
    }
    
  • 相关阅读:
    经典网络命令(搜集、概括)
    浅谈“五万月薪涉足数据恢复行业”
    C语言宏定义技巧(常用宏定义)
    安装IIS5.0出错
    IDM(Internet Download Manager)下载
    tape记忆法
    华为手环更换绑定手机
    冯况 | 清理电脑磁盘
    利用知网查个人信息
    双向循环链表
  • 原文地址:https://www.cnblogs.com/ivorysi/p/10108100.html
Copyright © 2011-2022 走看看