NOI2018 模拟 T2

http://www.elijahqi.win/archives/3885
题意:求选出树上k个点 使得sigma a[i]-a[i-1]最大 a[i]-a[i-1]表示两个点 树上距离

容易发现k=n的时候 所有边长*2-直径就是我们要的 那么k< n的时候我们类似虚树dp一下即可

设dp[i][j][0/1/2]表示当前在i子树选了j个点我子树内有0,1,2个直径的端点我的最小代价是多少

转移见方程~

#include
using namespace std;
inline char gc(){
    static char now[1<<16],*S,*T;
    if (T==S){T=(S=now)+fread(now,1,1<<16,stdin);if (T==S) return EOF;}
    return *S++;
}
inline int read(){
    int x=0,f=1;char ch=gc();
    while(!isdigit(ch)) {if (ch=='-') f=-1;ch=gc();}
    while(isdigit(ch)) x=x*10+ch-'0',ch=gc();
    return x*f;
}
const int N=3300;
struct node{
    int y,next,z;
}data[N<<1];
int dp[N][N][3],ans=0x3f3f3f3f,n,k,num,h[N],size[N];
inline void mn(int &x,int v){x=min(x,v);}
inline void dfs(int x,int fa){
    size[x]=1;dp[x][1][0]=dp[x][1][1]=0;
    for (int owo=h[x];owo;owo=data[owo].next){
        int y=data[owo].y;if (y==fa) continue;
        dfs(y,x);int z=data[owo].z;
        for (int i=size[x];~i;--i){
            for (int j=size[y];~j;--j){
                if (i+j>k) continue;
                mn(dp[x][i+j][0],dp[x][i][0]+dp[y][j][0]+(z<<1));
                mn(dp[x][i+j][1],dp[x][i][1]+dp[y][j][0]+(z<<1));
                mn(dp[x][i+j][1],dp[x][i][0]+dp[y][j][1]+z);
                mn(dp[x][i+j][2],dp[x][i][1]+dp[y][j][1]+z);
                mn(dp[x][i+j][2],dp[x][i][0]+dp[y][j][2]+(z<<1));
                mn(dp[x][i+j][2],dp[x][i][2]+dp[y][j][0]+(z<<1));
            }
        }size[x]+=size[y];
    }ans=min(ans,min(dp[x][k][0],min(dp[x][k][1],dp[x][k][2])));
}
int main(){
    freopen("2.in","r",stdin);
    n=read();k=read();memset(dp,0x3f,sizeof(dp));
    for (int i=1;iint x=read(),y=read(),z=read();
        data[++num].y=y;data[num].next=h[x];data[num].z=z;h[x]=num;
        data[++num].y=x;data[num].next=h[y];data[num].z=z;h[y]=num;
    }dfs(1,1);printf("%d\n",ans);
    return 0;
}

你可能感兴趣的:(动态规划)