bzoj 2427

tarjan缩点后即为根,然后就是简单的树型DP

然而DP花了我好长时间QAQ得找个时间练DP才行

 1 #include<bits/stdc++.h>
 2 #define inc(i,l,r) for(int i=l;i<=r;i++)
 3 #define dec(i,l,r) for(int i=l;i>=r;i--)
 4 #define link(x) for(edge *j=h[x];j;j=j->next)
 5 #define mem(a) memset(a,0,sizeof(a))
 6 #define inf 1e9
 7 #define ll long long
 8 #define succ(x) (1<<x)
 9 #define NM 500+5
10 using namespace std;
11 int read(){
12     int x=0,f=1;char ch=getchar();
13     while(!isdigit(ch)){if(ch==‘-‘)f=-1;ch=getchar();}
14     while(isdigit(ch))x=x*10+ch-‘0‘,ch=getchar();
15     return x*f;
16 }
17 struct edge{
18     int t,v;
19     edge *next;
20 }e[2*NM],*h[NM],*o=e;
21 void add(int x,int y){
22     o->t=y;o->next=h[x];h[x]=o;o++;
23 }
24 int n,m,a[NM],b[NM],_d[NM][NM],d[NM],suc[NM],low[NM],_x,_y,_t,tot,cnt,ans;
25 bool v[NM];
26 stack<int >s;
27 void dfs(int x){
28     d[x]=low[x]=++tot;s.push(x);
29     link(x)
30     if(!d[j->t]){
31         dfs(j->t);
32         low[x]=min(low[x],low[j->t]);
33     }else if(!suc[j->t])
34     low[x]=min(low[x],low[j->t]);
35     if(d[x]==low[x]){
36         int t;cnt++;
37         do{
38             t=s.top();s.pop();
39             suc[t]=cnt;
40         }while(x!=t);
41     }
42 }
43 void dp(int x){
44     inc(i,a[x],m)_d[x][i]=b[x];
45     link(x){
46         dp(j->t);
47         dec(k,m,a[x])
48         inc(i,a[j->t],k-a[x])
49         if(k>=i)
50         _d[x][k]=max(_d[x][k],_d[j->t][i]+_d[x][k-i]);
51     }
52 }
53 int main(){
54 //    freopen("data.in","r",stdin);
55     n=read();m=read();
56     inc(i,1,n)a[i]=read();
57     inc(i,1,n)b[i]=read();
58     inc(i,1,n)
59     if(_x=read())
60     add(_x,i),v[i]++;
61     inc(i,1,n)
62     if(!v[i])dfs(i);
63     inc(i,1,n)
64     if(!d[i])dfs(i);
65     inc(i,1,cnt){
66         _x=_y=_t=0;
67         inc(j,1,n)
68         if(suc[j]==i)_x+=a[j],_y+=b[j],_t++;
69         if(_t==1){
70             inc(j,1,n)
71             if(suc[j]==i&&!v[j])
72             add(0,j);
73         }else{
74             a[++n]=_x;b[n]=_y;
75             add(0,n);
76             inc(k,1,n)
77             if(suc[k]==i)
78             link(k)
79             if(suc[j->t]!=i)add(n,j->t);
80         }
81     }
82     dp(0);
83     inc(i,1,m)ans=max(ans,_d[0][i]);
84     printf("%d\n",ans);
85     return 0;
86 }

时间: 2024-10-10 23:33:38

bzoj 2427的相关文章

bzoj 2427: [HAOI2010]软件安装

Description 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和最大). 但是现在有个问题:软件之间存在依赖关系,即软件i只有在安装了软件j(包括软件j的直接或间接依赖)的情况下才能正确工作(软件i依赖软件j).幸运的是,一个软件最多依赖另外一个软件.如果一个软件不能正常工作,那么它能够发挥的作用为0. 我们现在知道了软件之间的依赖关系:软件i依赖软件Di.现在

BZOJ 2427 软件安装(强连通分量+树形背包)

题意:现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和最大).但是现在有个问题:软件之间存在依赖关系,即软件i只有在安装了软件j(包括软件j的直接或间接依赖)的情况下才能正确工作(软件i依赖软件j).幸运的是,一个软件最多依赖另外一个软件.如果一个软件不能正常工作,那么它能够发挥的作用为0. # include <cstdio> # include <cstri

BZOJ 2427: [HAOI2010]软件安装( dp )

软件构成了一些树和一些环, 对于环我们要不不选, 要么选整个环. 跑tarjan缩点后, 新建个root, 往每个入度为0的点(强连通分量) 连边, 然后跑树dp( 01背包 ) ---------------------------------------------------------------------------- #include<cstdio> #include<cstring> #include<stack> #include<algorit

bzoj 2427: [HAOI2010]软件安装【tarjan+树形dp】

一眼最大权闭合子图,然后开始构图,画了画之后发现我其实是个智障网络流满足不了m,于是发现正确的打开方式应该是一眼树上dp 然后仔细看了看性质,发现把依赖关系建成图之后是个奇环森林,这个显然不能直接dp 发现这个环要选的话只能选整个环,所以tarjan缩一下点,然后再跑树上背包就行了 #include<iostream> #include<cstdio> #include<vector> #include<cstring> using namespace st

[Bzoj 2427] [HAOI2010] 软件安装 tarjan缩点+树形DP

题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一 些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和最大). 但是现在有个问题:软件之间存在依赖关系,即软件i只有在安装了软件j(包括软件j的直接或间接依赖)的情况下才能正确工作(软件i依赖软件j).幸运的 是,一个软件最多依赖另外一个软件.如果一个软件不能正常工作,那么它能够发挥的作用为0. 我们现在知道了软件之间的依赖关系:软件i依赖软件Di.现在请你设计出

BZOJ 1013: [JSOI2008]球形空间产生器sphere

二次联通门 : BZOJ 1013: [JSOI2008]球形空间产生器sphere /* BZOJ 1013: [JSOI2008]球形空间产生器sphere 高斯消元 QAQ SB的我也能终于能秒题了啊 设球心的坐标为(x,y,z...) 那么就可以列n+1个方程,化化式子高斯消元即可 */ #include <cstdio> #include <iostream> #include <cstring> #define rg register #define Max

bzoj 3309 DZY Loves Math - 莫比乌斯反演 - 线性筛

对于正整数n,定义f(n)为n所含质因子的最大幂指数.例如f(1960)=f(2^3 * 5^1 * 7^2)=3, f(10007)=1, f(1)=0. 给定正整数a,b,求sigma(sigma(f(gcd(i,j)))) (i=1..a, j=1..b). Input 第一行一个数T,表示询问数. 接下来T行,每行两个数a,b,表示一个询问. Output 对于每一个询问,输出一行一个非负整数作为回答. Sample Input 4 7558588 9653114 6514903 445

【BZOJ】[HNOI2009]有趣的数列

[算法]Catalan数 [题解] 学了卡特兰数就会啦>_<! 因为奇偶各自递增,所以确定了奇偶各自的数字后排列唯一. 那么就是给2n个数分奇偶了,是不是有点像入栈出栈序呢. 将做偶数标为-1,做奇数标为+1,显然当偶数多于奇数时不合法,因为它压不住后面的奇数. 然后其实这种题目,打表就可知啦--QAQ 然后问题就是求1/(n+1)*C(2n,n)%p了,p不一定是素数. 参考bzoj礼物的解法. 看到网上清一色的素数筛+分解质因数解法,不解了好久,感觉写了假的礼物-- 后来觉得礼物的做法才比

洛谷 P2709 BZOJ 3781 小B的询问

题目描述 小B有一个序列,包含N个1~K之间的整数.他一共有M个询问,每个询问给定一个区间[L..R],求Sigma(c(i)^2)的值,其中i的值从1到K,其中c(i)表示数字i在[L..R]中的重复次数.小B请你帮助他回答询问. 输入输出格式 输入格式: 第一行,三个整数N.M.K. 第二行,N个整数,表示小B的序列. 接下来的M行,每行两个整数L.R. 输出格式: M行,每行一个整数,其中第i行的整数表示第i个询问的答案. 输入输出样例 输入样例#1: 6 4 3 1 3 2 1 1 3