zoukankan      html  css  js  c++  java
  • 【JZOJ3347】树的难题【树形dp】【拓扑排序】

    题目大意:

    题目链接:https://jzoj.net/senior/#main/show/3347
    在这里插入图片描述


    思路:

    f[x]f[x]表示以xx为根的子树中,满足有0个黑色点,若干个白色点的最少切割次数;g[x]g[x]表示以xx为根的子树中,满足有若干个黑色点,0个白色点的最少切割次数;h[x]h[x]表示以xx为根的子树中,满足有若干个黑色点,1个白色点的最少切割次数。iixx的儿子。

    • 如果xx是黑色的
      1. f[x]f[x]需要满足没有黑色节点,显然不满足。
      2. g[x]=imin(f[v]+e[i].dis,g[v],h[v]+e[i].dis)g[x]=sum^i min(f[v]+e[i].dis,g[v],h[v]+e[i].dis)
      3. h[x]=min(h[u],h[v]+g[u]minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis))h[x]=min(h[u],h[v]+g[u]-minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis))
    • 如果xx是白色的
      1. f[x]=imin(f[v],g[v]+e[i].dis,h[v]+e[i].dis)f[x]=sum^i min(f[v],g[v]+e[i].dis,h[v]+e[i].dis)
      2. g[x]g[x]显然不满足
      3. h[x]=imin(f[v]+e[i].dis,g[v],h[v]+e[i].dis)h[x]=sum^i min(f[v]+e[i].dis,g[v],h[v]+e[i].dis)
    • 如果xx是灰色的
      1. f[x]=imin(f[v],g[v]+e[i].dis,h[v]+e[i].dis)f[x]=sum^i min(f[v],g[v]+e[i].dis,h[v]+e[i].dis)
      2. g[x]=imin(f[v]+e[i].dis,g[v],h[v]+e[i].dis)g[x]=sum^i min(f[v]+e[i].dis,g[v],h[v]+e[i].dis)
      3. h[x]=min(h[u],h[v]+g[u]minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis))h[x]=min(h[u],h[v]+g[u]-minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis))

    还是比较好推的。毕竟除了设状态以外都是自己推的。
    代码又长又丑,恶心死了。听说普通的dpdp会爆栈,还敲了一个topsorttopsortbfsbfsdpdp,而且topsorttopsortbfsbfs还是一起进行的。。。
    时间复杂度O(Tn)O(Tn)


    代码:

    #include <queue>
    #include <cstdio>
    #include <string>
    #include <cstring>
    using namespace std;
    typedef long long ll;
    
    const int N=300010;
    const ll Inf=1e17;
    int T,n,tot,root,head[N],col[N],in[N];
    ll f[N],g[N],h[N];
    
    struct edge
    {
    	int next,to;
    	ll dis;
    }e[N*2];
    
    int read()
    {
    	int d=0;
    	char ch=getchar();
    	while (!isdigit(ch)) ch=getchar();
    	while (isdigit(ch))
    		d=(d<<3)+(d<<1)+ch-48,ch=getchar();
    	return d;
    }
    
    ll minn(ll x,ll y,ll z)
    {
    	return min(x,min(y,z));
    }
    
    void add(int from,int to,ll dis)
    {
    	e[++tot].to=to;
    	e[tot].dis=dis;
    	e[tot].next=head[from];
    	head[from]=tot;
    }
    
    void dp()
    {
    	queue<int> q;
    	for (int i=1;i<=n;i++)
    		if (in[i]==1) q.push(i);
    	while (q.size())
    	{
    		int u=q.front();
    		q.pop();
    		root=u;
    		in[u]=0;
    		
    		if (col[u]==0)
    		{
    			f[u]=Inf;
    			for (int i=head[u];~i;i=e[i].next)
    			{
    				int v=e[i].to;
    				if (!in[v]) g[u]+=minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis);
    			}
    			for (int i=head[u];~i;i=e[i].next)
    			{
    				int v=e[i].to;
    				if (!in[v]) h[u]=min(h[u],h[v]+g[u]-minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis));
    			}
    		}
    		
    		if (col[u]==1)
    		{
    			g[u]=Inf;
    			for (int i=head[u];~i;i=e[i].next)
    			{
    				int v=e[i].to;
    				if (!in[v])
    				{
    					f[u]+=minn(f[v],g[v]+e[i].dis,h[v]+e[i].dis);
    					h[u]+=minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis);
    				}
    			}
    		}
    		
    		if (col[u]==2)
    		{
    			for (int i=head[u];~i;i=e[i].next)
    			{
    				int v=e[i].to;
    				if (!in[v])
    				{
    					f[u]+=minn(f[v],g[v]+e[i].dis,h[v]+e[i].dis);
    					g[u]+=minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis);
    				}
    			}
    			for (int i=head[u];~i;i=e[i].next)
    			{
    				int v=e[i].to;
    				if (!in[v]) h[u]=min(h[u],h[v]+g[u]-minn(f[v]+e[i].dis,g[v],h[v]+e[i].dis));
    			}
    		}
    		
    		for (int i=head[u];~i;i=e[i].next)
    		{
    			int v=e[i].to;
    			if (in[v]>1)
    			{
    				in[v]--;
    				if (in[v]==1) q.push(v);
    			}
    		}
    	}
    }
    
    int main()
    {
    	T=read();
    	while (T--)
    	{
    		memset(head,-1,sizeof(head));
    		memset(in,0,sizeof(in));
    		tot=0;
    		n=read();
    		for (int i=1;i<=n;i++)
    		{
    			col[i]=read();
    			if (col[i]==0) f[i]=Inf,g[i]=0,h[i]=Inf;
    			if (col[i]==1) f[i]=0,g[i]=Inf,h[i]=0;
    			if (col[i]==2) f[i]=0,g[i]=0,h[i]=Inf;
    		}
    		for (int i=1,x,y,z;i<n;i++)
    		{
    			x=read(); y=read(); z=read();
    			add(x,y,(ll)z); add(y,x,(ll)z);
    			in[x]++; in[y]++;
    		}
    		dp();
    		printf("%lld
    ",minn(f[root],g[root],h[root]));
    	}
    	return 0;
    }
    
  • 相关阅读:
    Go语言
    Go语言
    electron-builder vue3 用户自定义安装目录
    提取页面中的style标签内容
    px2rpx | px转rpx
    js EventBus
    select 下拉选择多个值
    keep-alive页面缓存
    js适配移动端页面
    vue日常问题记录
  • 原文地址:https://www.cnblogs.com/hello-tomorrow/p/11998092.html
Copyright © 2011-2022 走看看