zoukankan      html  css  js  c++  java
  • Codeforces 868F Yet Another Minimization Problem(分治+莫队优化DP)

    题目链接  Yet Another Minimization Problem

    题意  给定一个序列,现在要把这个序列分成k个连续的连续子序列。求每个连续子序列价值和的最小值。

    设$f[i][j]$为前$i$个数分成$j$段的最优解

    不难得出状态转移方程$f[i][j] = min(f[k][j - 1], calc(j + i, i))$

    该DP具有决策单调性

    即若$f[i][j]$是从$f[x][j - 1]$转移到的,$f[i+1][j]$是从$f[y][j - 1]$转移到的,那么一定有$x <= y$。

    考虑到这一点就可以用分治优化。

    还有一点就是$calc()$的计算。

    用莫队计算就可以了(分治的时候同一个递归状态下莫队查询端点的改变都是连续的)

    时间复杂度$O(nklogn)$

    #include <bits/stdc++.h>
    
    using namespace std;
    
    #define rep(i, a, b)	for (int i(a); i <= (b); ++i)
    #define dec(i, a, b)	for (int i(a); i >= (b); --i)
    
    typedef long long LL;
    
    const int N = 1e5 + 10;
    const int A = 22;
    
    LL f[N][A], ret;
    int a[N], cnt[N];
    int n, m, l, r;
    
    void query_init(){
    	memset(cnt, 0, sizeof cnt);
    	l = 1, r = 0;
    	ret = 0;
    }
    
    LL query(int ql, int qr){
    	while (r < qr){
    		++r;
    		ret += 1ll * cnt[a[r]];
    		++cnt[a[r]];
    	}
    
    	while (r > qr){
    		--cnt[a[r]];
    		ret -= 1ll * cnt[a[r]];
    		--r;
    	}
    
    	while (l > ql){
    		--l;
    		ret += 1ll * cnt[a[l]];
    		++cnt[a[l]];
    	}
    
    	while (l < ql){
    		--cnt[a[l]];
    		ret -= 1ll * cnt[a[l]];
    		++l;
    	}
    
    	return ret;
    }
    
    void solve(int j, int l, int r, int st, int ed){
    	if (l > r) return;
    	int mid = (l + r) >> 1;
    	int x;
    
    	rep(i, st, min(mid, ed)){
    		LL now = query(i, mid);
    		if (f[i - 1][j - 1] + now <= f[mid][j]){
    			f[mid][j] = f[i - 1][j - 1] + now;
    			x = i;
    		}
    	}
    
    	if (l != r){
    		solve(j, l, mid - 1, st, x);
    		solve(j, mid + 1, r, x, ed);
    	}
    }
    
    int main(){
    
    	scanf("%d%d", &n, &m);
    	rep(i, 1, n) scanf("%d", a + i);
    
    	query_init();
    
    	rep(i, 1, n) rep(j, 0, m) f[i][j] = 1e18;
    	rep(i, 1, n) f[i][1] = query(1, i);
    	rep(j, 2, m) solve(j, 1, n, 1, n);
    
    	printf("%lld
    ", f[n][m]);
    	return 0;
    }
    
  • 相关阅读:
    poj-2478 Farey Sequence(dp,欧拉函数)
    codeforces 515C C. Drazil and Factorial(水题,贪心)
    hdu-5643 King's Game(打表)
    可以在命令行直接使用密码来进行远程连接和远程拉取文件的命令:sshpass
    查看SQL运行时间
    管理kafka
    No module named _sqlite3
    ps
    Redis之RDB与AOF
    Redis 4 参数解释
  • 原文地址:https://www.cnblogs.com/cxhscst2/p/7644200.html
Copyright © 2011-2022 走看看