알고리즘/SW Expert Academy
[SWEA] 4534 트리 흑백 색칠
세진짱
2020. 3. 2. 18:28
dp를 통해 색칠해 나갈 수 있다
우선 흰색 검은색 가능한데 동시에 연결된 정점이 (검은색/검은색)은 불가능하다
그럼 이 부분만 잘 처리하면 dfs+dp를 통해 해결할 수 있다!
먼저 나의 색을 저장하고 내가 만약 흰색이라면
내 자식들을 (검은색으로 칠하는 경우 + 흰색으로 칠하는 경우)만큼 가능하고
검은색이라면 (흰색으로 칠하는 경우)만 가능하다
경우의 수 이기 때문에 각 자식들의 수의 곱을 구해가면 된다!
long long 조심하자~~
소스코드
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
#include <bits/stdc++.h>
using namespace std;
#define ll long long
const int MAXN = 1e5+5;
const ll mod=1e9+7LL;
ll d[MAXN][2];
int n;
ll solve(vector<vector<int>>&tree, int pos,int color,int parent){
ll&ret = d[pos][color];
if(ret != -1) return ret;
ret=1;
for(int i=0;i<tree[pos].size();i++){
int next = tree[pos][i];
if(next==parent) continue;
if(!color) ret*=(solve(tree,next,0,pos)+solve(tree,next,1,pos));
else ret*=solve(tree,next,0,pos);
ret%=mod;
}
return ret;
}
int main(){
int t; scanf(" %d",&t);
for(int tc=1;tc<=t;tc++){
memset(d,-1,sizeof(d));
scanf(" %d",&n);
vector<vector<int>> tree(n);
for(int i=0;i<n-1;i++){
int x,y; scanf(" %d %d",&x,&y);
x--;y--;
tree[x].push_back(y);
tree[y].push_back(x);
}
ll ans = solve(tree,0,0,-1)%mod+solve(tree,0,1,-1)%mod;
ans%=mod;
printf("#%d %lld\n",tc,ans);
}
}
|