【xsy1120】 支援(assist) dp+卡常

妙啊算错时间复杂度了

题目大意:给你一棵$n$个节点的二叉树,每个节点要么是叶子节点,要么拥有恰好两个儿子。

令$m$为叶子节点个数,你需要在这棵二叉树中选择$i$个叶子节点染色,叶节点染色需要一定的代价,非叶子节点代价为两孩子的染色节点数量的异或和乘上一常数。请最小化代价。

数据范围:$n≤4000$。

 

显然这是一道$dp$题。

令$f[u][i]$表示在以$u$号点为根的子树中,选择$i$个叶子节点染色的最小代价。

若u为叶子节点,不难得出$f[u][0]=0$,$f[u][1]=c[u]$。

我们不难得出$f[u][i+j]=min{f[lc][i]+f[rc][j]+c[u]*(i\ xor\ j)}$,其中$lc$,$rc$表示$u$的左儿子和右儿子,$c[u]$表示$u$号节点的常数。

然后这个$dp$转移,看似是$O(n^3)$的,实际上:

$T(n)=2T(\frac{n}{2})+O(n^2)$。这个实际上还是$O(n^2)$的。。。。。。。。

然后就愉快地做完了,注意卡常。

 

 1 #include<bits/stdc++.h>
 2 #define M 4005
 3 #define L long long
 4 #define INF (1<<28)
 5 using namespace std;
 6 int l[M]={0},r[M]={0},siz[M]={0},c[M]={0},vis[M]={0},f[M][M]={0};
 7 
 8 void dfs(int x){
 9     if(vis[x]) return; vis[x]=1;
10     if(l[x]==0&&r[x]==0){
11         siz[x]=1;
12         f[x][0]=0; f[x][1]=c[x];
13         return;
14     }
15     dfs(l[x]); dfs(r[x]);
16     siz[x]=siz[l[x]]+siz[r[x]];
17     for(int i=0;i<=siz[l[x]];i++){
18         for(int j=0;j<=siz[r[x]];j++)
19         f[x][i+j]=min(f[x][i+j],f[l[x]][i]+f[r[x]][j]+c[x]*(i^j));
20     }
21 }
22 
23 int Main(){
24     memset(vis,0,sizeof(vis));
25     int n; scanf("%d",&n);
26     for(int i=1;i<=n;i++) scanf("%d%d",l+i,r+i);
27     for(int i=1;i<=n;i++) scanf("%d",c+i);
28     for(int i=1;i<=n;i++) memset(f[i],63,(n+1)<<2);
29     for(int i=1;i<=n;i++) dfs(i);
30     int rt=0; for(int i=1;i<=n;i++) if(siz[rt]<siz[i]) rt=i;
31     for(int i=1;i<=siz[rt];i++)
32         printf("%d ",f[rt][i]); 
33     printf("\n");
34 }
35 int main(){
36     int cas; cin>>cas;
37     while(cas--) Main();
38 }
posted @ 2018-10-20 09:57  AlphaInf  阅读(184)  评论(2编辑  收藏  举报