NOI2018 模拟 T2
Posted wx62f0894128448
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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<bits/stdc++.h>
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;i<n;++i)
int 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;
以上是关于NOI2018 模拟 T2的主要内容,如果未能解决你的问题,请参考以下文章