P4084 [USACO17DEC] Barn Painting G 题解

题目描述

Farmer John 有一个大农场,农场上有 N 个谷仓(1≤N≤105),其中一些已经涂色,另一些尚未涂色。Farmer John 想要为这些剩余的谷仓涂色,使得所有谷仓都被涂色,但他只有三种可用的油漆颜色。此外,他的获奖奶牛 Bessie 如果发现两个直接相连的谷仓颜色相同,会感到困惑,因此他希望确保这种情况不会发生。

保证 N 个谷仓之间的连接不会形成任何"环"。也就是说,任意两个谷仓之间最多只有一条连接路径。

Farmer John 有多少种方式可以为剩余的未涂色谷仓涂色?

输入格式

第一行包含两个整数 N 和 K(0≤K≤N),分别表示农场上的谷仓数量和已经涂色的谷仓数量。

接下来的 N−1 行每行包含两个整数 x 和 y(1≤x,y≤N,x=y),描述直接连接谷仓 x 和 y 的路径。

接下来的 K 行每行包含两个整数 b 和 c(1≤b≤N, 1≤c≤3),表示谷仓 b 已经被涂成颜色 c。

输出格式

计算为剩余谷仓涂色的有效方式数量,模 109+7,要求任何两个直接相连的谷仓颜色不同。

输入输出样例

输入 #1

复制代码
4 1
1 2
1 3
1 4
4 3

输出 #1

复制代码
8

思路:

依旧树形DP。

状态:dpi0代表i涂第1种颜色的方法数,dpi1代表i涂第2种颜色的方法数,dpi2代表i涂第3种颜色的方法数。

状态转移方程:

dpu0=dpu0*(dpv2+dpv1)%mod;
dpu1=dpu1*(dpv2+dpv0)%mod;
dpu2=dpu2*(dpv1+dpv0)%mod;

代码:

cpp 复制代码
#include <bits/stdc++.h>
#include <bits/c++config.h>
#include <ostream>
#include <istream>
#include <algorithm>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <string>
#include <math.h>
#include <time.h>
#include <ctime>
#include <cstdlib>

#define ll long long
#define ull unsigned long long
#define db double
#define st string
#define ch char
#define bo bool
#define s1 27
#define s2 205
#define s3 2005
#define s4 20005
#define s5 200005
#define s6 2000005
#define s7 20000005
using namespace std;
ll mod=1000000007;
ll n,m,dp[s5][3],a[s5];
//dp[i][0]代表i涂第1种颜色,dp[i][1]代表i涂第2种颜色,dp[i][2]代表i涂第3种颜色。
vector<int> g[s5];
void dfs(int u,int fa){
    if(a[u]==0){
    	for(int i=0;i<g[u].size();i++){
    		int v=g[u][i];
            if(v==fa) continue;
    		dfs(v,u);
            //dp[u][0]
            dp[u][0]=dp[u][0]*(dp[v][2]+dp[v][1])%mod;
            //dp[u][1]
            dp[u][1]=dp[u][1]*(dp[v][2]+dp[v][0])%mod;
            //dp[u][2]
            dp[u][2]=dp[u][2]*(dp[v][1]+dp[v][0])%mod;
    	}
    }
    else if(a[u]!=0){
        if(a[u]==1){
            dp[u][1]=0;
            dp[u][2]=0;
            for(int i=0;i<g[u].size();i++){
	    		int v=g[u][i];
	            if(v==fa) continue;
	    		dfs(v,u);
	            dp[u][0]=dp[u][0]*(dp[v][2]+dp[v][1])%mod;
	    	}
        }
        else if(a[u]==2){
            dp[u][2]=0;
            dp[u][0]=0;
            for(int i=0;i<g[u].size();i++){
	    		int v=g[u][i];
	            if(v==fa) continue;
	    		dfs(v,u);
	            dp[u][1]=dp[u][1]*(dp[v][2]+dp[v][0])%mod;
	    	}
        }
        else if(a[u]==3){
            dp[u][1]=0;
            dp[u][0]=0;
            for(int i=0;i<g[u].size();i++){
	    		int v=g[u][i];
	            if(v==fa) continue;
	    		dfs(v,u);
	            dp[u][2]=dp[u][2]*(dp[v][1]+dp[v][0])%mod;
	    	}
        }
    }
    return ;
}
signed main(){
	cin>>n>>m;
    for(int i=0;i<=100005;i++){
        dp[i][1]=1;
        dp[i][0]=1;
        dp[i][2]=1;
    }
	for(int i=1;i<n;i++){
		int k,c;
		cin>>k>>c;
		g[c].push_back(k);
        g[k].push_back(c);
	}
    for(int i=1;i<=m;i++){
        int b,c;
        cin>>b>>c;
        a[b]=c;
    }
	dfs(1,0);
    cout<<(dp[1][0]+dp[1][1]%mod+dp[1][2]%mod)%mod<<endl;
	return 0;
}
相关推荐
2401_872418786 小时前
算法入门:数据结构-堆
数据结构·算法
王老师青少年编程6 小时前
信奥赛C++提高组csp-s之搜索进阶(搜索剪枝案例实践1)
c++·csp·高频考点·信奥赛·提高组·搜索剪枝·小木棍
xwz小王子7 小时前
手术机器人登上Science Robotics:2毫米纤细手臂,从3厘米切口完成腰椎神经减压
算法·机器人
黎阳之光8 小时前
视频孪生智护供水生命线:黎阳之光赋能医疗与园区水务高质量升级
运维·物联网·算法·安全·数字孪生
Black蜡笔小新8 小时前
自动化AI算法训练服务器DLTM制造业AI质检工作站助力制造业实现AI智检
人工智能·算法·自动化
嵌入式小能手8 小时前
飞凌嵌入式ElfBoard-进程间的通信之命名管道
linux·服务器·算法
啦哈拉哈9 小时前
Leetcode题解记录-hot100(81-100)
算法·leetcode·职场和发展
csdn_aspnet9 小时前
Java 霍尔分区算法(Hoare‘s Partition Algorithm)
java·开发语言·算法
王老师青少年编程9 小时前
信奥赛C++提高组csp-s之搜索进阶(搜索剪枝核心思想 )
c++·dfs·csp·信奥赛·搜索剪枝·搜索优化
一拳一个呆瓜9 小时前
【STL】使用 C++ 标准库标头
c++·stl