5.21 省选模拟赛 luogu P4297 [NOI2006]网络收费 树形dp

LINK:网络收费

还是自己没脑子. 早上思考的时候 发现树形dp不可做 然后放弃治疗了.

没有合理的转换问题的模型是我整个人最大的败笔.

暴力也值得一提 爆搜之后可以写成FFT的形式的计算贡献的方法 连图都不用建出来.

不是传统的树形dp 因为子树的状态影响之后的决策 并且从下至上的话需要状压所有点的状态 从上之下的话代价难以统计。

观察图中的这张表格 容易发现有规律的事情 当 na<nb时 有A的一定付出代价 两个A的话就两倍 一个A的话就一倍 B的话不要代价。

容易转换成上述模型 于是 这个点的颜色还是为 子树内A和B谁多的问题。

还是考虑树形dp 考虑从下至上 发现还是难以统计答案。从上之下 发现可以统计答案了 不过存在不合法 对于不合法的状态扔掉即可。

说起来这类似于爆搜 只搜路径上的点的状态 然后进行合并 由于前面的状态被搜过 后面保证所有状态也被搜了一遍 所以可以发现爆搜出了所有的状态.

合并的时候 其实类似背包。非常巧妙的题目。

const int MAXN=2100;
int n,m,maxx,ans=INF,top;
int w[MAXN][MAXN],s[MAXN],q[MAXN],vis[MAXN],a[MAXN],c[MAXN],b[MAXN][MAXN];
int f[MAXN][MAXN],sz[MAXN];//f[i][j]表示以i为根的子树内B的点数为j的最小代价.
inline int LCA(int x,int y)
{
	while(x!=y)
	{
		x>>=1;
		y>>=1;
	}
	return x;
}
inline void dfs(int x,int dep)
{
	rep(0,sz[x],j)f[x][j]=INF;
	if(dep==n)
	{
		f[x][0]=f[x][1]=0;
		fep(top,1,i)
		{
			if(s[i]!=a[x])f[x][a[x]]+=w[x][q[i]];
			else f[x][a[x]^1]+=w[x][q[i]];
		}
		f[x][a[x]^1]+=c[x];
		return;
	}
	s[++top]=1;q[top]=x;
	dfs(x<<1,dep+1);
	dfs(x<<1|1,dep+1);
	rep(0,sz[x<<1],i)rep(0,sz[x<<1|1],j)
		if(i+j>sz[x]-i-j)f[x][i+j]=min(f[x][i+j],f[x<<1][i]+f[x<<1|1][j]);
	s[top]=0,q[top]=x;
	dfs(x<<1,dep+1);
	dfs(x<<1|1,dep+1);
	rep(0,sz[x<<1],i)rep(0,sz[x<<1|1],j)if(i+j<=sz[x]-i-j)f[x][i+j]=min(f[x][i+j],f[x<<1][i]+f[x<<1|1][j]);
	--top;
}
int main()
{
	freopen("1.in","r",stdin);
	get(n);m=1<<n;maxx=(1<<n+1)-1;
	rep(m,maxx,i)get(a[i]),sz[i]=1;
	rep(m,maxx,i)get(c[i]);
	rep(m,maxx,i)rep(i+1,maxx,j)get(b[i][j]);
	rep(m,maxx,i)rep(i+1,maxx,j)
	{
		int lca=LCA(i,j);
		w[i][lca]+=b[i][j];
		w[j][lca]+=b[i][j];
	}
	fep(maxx,1,i)if((i<<1|1)<=maxx)sz[i]=sz[i<<1|1]+sz[i<<1];
	dfs(1,0);rep(0,sz[1],i)ans=min(ans,f[1][i]);put(ans);return 0;
}
posted @ 2020-05-21 19:12  chdy  阅读(117)  评论(0编辑  收藏  举报