Codeforces 868F. Yet Another Minimization Problem

Posted yuzao

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Codeforces 868F. Yet Another Minimization Problem相关的知识,希望对你有一定的参考价值。

Description

给出一个长度为 (n) 的序列,你需要将它分为 (k) 段,使得每一段的价值和最小,每一段的价值是这一段内相同的数的个数
题面

Solution

容易想到设 (f[i][j]) 表示前 (i) 个数分成 (j) 段的最小代价
(f[i][j]=min(f[k][j-1]+w(k+1,i)))
这个(DP)有决策单调性,可以分治优化
(solve(l,r,L,R)) 表示用区间 ([L,R]) 内的决策去更新 ([l,r]) 的函数,找到 (mid) 的决策点然后分治下去就可以了

对于 (w) 函数的处理还需要注意:
我们维护两个全局指针,每一次移动到指定的区间,这样就可以 (O(1)) 转移了,均摊指针的移动次数是 (O(n*log))

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=1e5+10;
int n,a[N],m,t[N],p=1,q=0;ll f[N][22],tot=0;
inline void upd(int l,int r){
    while(q<r)q++,tot+=t[a[q]],t[a[q]]++;
    while(p>l)p--,tot+=t[a[p]],t[a[p]]++;
    while(q>r)t[a[q]]--,tot-=t[a[q]],q--;
    while(p<l)t[a[p]]--,tot-=t[a[p]],p++;
}
inline void solve(int l,int r,int L,int R,int k){
    if(l==r){
        for(int i=min(R,r);i>=L;i--)
            upd(i+1,l),f[l][k]=min(f[l][k],f[i][k-1]+tot);
        return ;
    }
    int mid=(l+r)>>1,st=0;
    for(int i=min(mid-1,R);i>=L;i--){
        upd(i+1,mid);
        if(tot+f[i][k-1]<f[mid][k])f[mid][k]=tot+f[i][k-1],st=i;
    }
    if(st)solve(l,mid,L,st,k),solve(mid+1,r,st,R,k);
    else solve(mid+1,r,L,R,k);
}
int main(){
  freopen("pp.in","r",stdin);
  freopen("pp.out","w",stdout);
  scanf("%d%d",&n,&m);
  for(int i=1;i<=n;i++)scanf("%d",&a[i]);
  memset(f,127/3,sizeof(f));f[0][1]=0;
  for(int i=1;i<=n;i++)f[i][1]=f[i-1][1]+t[a[i]],t[a[i]]++;
  memset(t,0,sizeof(t));
  for(int i=2;i<=m;i++)
      solve(1,n,1,n,i);
  cout<<f[n][m]<<endl;
  return 0;
}

以上是关于Codeforces 868F. Yet Another Minimization Problem的主要内容,如果未能解决你的问题,请参考以下文章

Codeforces 868F. Yet Another Minimization Problem决策单调性优化DP分治莫队

CodeForces - 868F Yet Another Minimization Problem

CodeForces 868F Yet Another Minimization Problem(决策单调性优化 + 分治)

cf868F. Yet Another Minimization Problem(决策单调性 分治dp)

[CF868F] Yet Another Minimization Problem

CF868 F. Yet Another Minimization Problem 决策单调优化 分治