HDU - 6643: Ridiculous Netizens(点分治+依赖背包+空间优化)
题意:给定带点权的树,问多少个连通块,其乘积<=M; N<=2000,M<1e6;
思路:连通块-->分治; 由于普通的树DP在合并的时候复杂度会高一个M,所以用依赖背包来做。 (当然,由于体积分布是离散的,可能有些选手用map也可以过,这样避免了每次都for(i,1,M),取决于数据吧)。
那么现在的复杂度就是O(NlogN*M) ,空间为O(N*M),尚待优化。
这里非常巧妙的把<sqrt(M)的和大于sqrt(M)的分开保存,那么前者就是正常的背包,表示背包里存了多少东西; 后者可以看成背包里最多还可以存多少东西。 那么复杂度就变成了O(Nsqrt(M)logN); 空间O(Nsqrt(M)); 就可以过了。
#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define ll long long
using namespace std;
const int maxn=;
const int Mod=1e9+;
int w[maxn],ans;
int dp1[maxn][maxn>>],dp2[maxn][maxn>>],dp[maxn][maxn];
int Laxt[maxn],Next[maxn<<],To[maxn<<],cnt;
int son[maxn],sz[maxn],vis[maxn],rt,SZ;
int times,p[maxn],M,qM;
void MOD(int &X){if(X>Mod) X-=Mod;}
void add(int u,int v)
{
Next[++cnt]=Laxt[u]; Laxt[u]=cnt; To[cnt]=v;
}
void getroot(int u,int f) //得到重心
{
sz[u]=; son[u]=;
for(int i=Laxt[u];i;i=Next[i]){
int v=To[i]; if(vis[v]||v==f) continue;
getroot(v,u);
sz[u]+=sz[v];
son[u]=max(son[u],sz[v]);
}
son[u]=max(son[u],SZ-son[u]);
if(rt==||son[u]<son[rt]) rt=u;
}
void dfs(int u,int f) //得到dfs序。
{
p[++times]=u; sz[u]=;
for(int i=Laxt[u];i;i=Next[i]){
if(To[i]==f||vis[To[i]]) continue;
dfs(To[i],u);
sz[u]+=sz[To[i]];
}
}
void cal()
{
rep(i,,times+){
memset(dp1[i],,sizeof(dp1[i]));
memset(dp2[i],,sizeof(dp2[i]));
}
dp1[times+][]=;
for(int i=times;i>=;i--){
int x=w[p[i]];
rep(j,,min(qM,M/x)) {
int k=j*x;
if(k<=qM) MOD(dp1[i][k]+=dp1[i+][j]);
else MOD(dp2[i][M/k]+=dp1[i+][j]);
}
rep(j,x,qM) {
MOD(dp2[i][j/x]+=dp2[i+][j]);
}
rep(j,,qM) MOD(dp1[i][j]+=dp1[i+sz[p[i]]][j]);
rep(j,,qM) MOD(dp2[i][j]+=dp2[i+sz[p[i]]][j]);
}
rep(i,,qM) MOD(ans+=dp1[][i]);
rep(i,,qM) MOD(ans+=dp2[][i]);
ans--; //减去为空的情况
if(ans<) ans+=Mod;
}
void solve(int u) //分治
{
vis[u]=;
times=; dfs(u,);
cal();
for(int i=Laxt[u];i;i=Next[i]){
if(vis[To[i]]) continue;
SZ=sz[To[i]]; rt=;
getroot(To[i],);
solve(rt);
}
}
int main()
{
int T,N,u,v;
scanf("%d",&T);
while(T--){
scanf("%d%d",&N,&M); qM=sqrt(M);
rep(i,,N) scanf("%d",&w[i]);
rep(i,,N) Laxt[i]=vis[i]=; cnt=;
rep(i,,N-){
scanf("%d%d",&u,&v);
add(u,v); add(v,u);
}
SZ=N; rt=; getroot(,);
ans=; solve(rt);
printf("%d\n",ans);
}
return ;
}
HDU - 6643: Ridiculous Netizens(点分治+依赖背包+空间优化)的更多相关文章
- Hdu 6268 点分治 树上背包 bitset 优化
给你一颗大小为n(3000)的树,树上每个点有点权(100000),再给你一个数m(100000) i为1~m,问树中是否存在一个子图,使得权值为i. 每次solve到一个节点 用一个bitset维护 ...
- [HDU多校]Ridiculous Netizens
[HDU多校]Ridiculous Netizens 点分治 分成两个部分:对某一点P,连通块经过P或不经过P. 经过P采用树形依赖背包 不经过P的部分递归计算 树型依赖背包 v点必须由其父亲u点转移 ...
- 依赖背包优化——hdu1561
傻逼依赖背包的优化 #include<bits/stdc++.h> using namespace std; #define N 205 ]; int head[N],tot,n,m,a[ ...
- 依赖背包变形——poj1947(经典)
/*这题显然不适用依赖背包的优化,因为不能保证根是必选的,但是可以按照常规依赖背包的思路进行转移,即每次对一个儿子进行C^2的转移 还是树形的背包,dp[u][j]表示u的子树里,切割出一个大小为j的 ...
- 【HDU 4276】The Ghost Blows Light(树形DP,依赖背包)
The Ghost Blows Light Problem Description My name is Hu Bayi, robing an ancient tomb in Tibet. The t ...
- BZOJ.4182.Shopping(点分治/dsu on tree 树形依赖背包 多重背包 单调队列)
BZOJ 题目的限制即:给定一棵树,只能任选一个连通块然后做背包,且每个点上的物品至少取一个.求花费为\(m\)时最大价值. 令\(f[i][j]\)表示在点\(i\),已用体积为\(j\)的最大价值 ...
- hdu 1561 The more, The Better (依赖背包 树形dp)
题目: 链接:点击打开链接 题意: 非常明显的依赖背包. 思路: dp[i][j]表示以i为根结点时攻击j个城堡得到的最大值.(以i为根的子树选择j个点所能达到的最优值) dp[root][j] = ...
- HDU 3449 依赖背包
这道题虽然水水的,但是还是成功地给我增加了10多个WA. 最开始拿着题,一看,依赖背包嘛~直接DFS树形DP嗨起来,甚至连内存都没有算一下,3MLE: 然后又仔细看了一下题,没有必要用树形背包来做嘛, ...
- HDU 2159 FATE(二维费用背包)
FATE Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submi ...
随机推荐
- 【java】获取客户端访问的公网ip和归属地
import com.alibaba.druid.support.json.JSONUtils; import org.thymeleaf.util.StringUtils; import javax ...
- angular JS中 ‘=’与angular.copy的区别
先来看代码: <b>{{test1}}</b> <input type="text" ng-model="test2" title ...
- golang gin 上传图片到aws s3
要上传图片到aws s3首先需要 知道 aws 的地区 也就是region ,还需要知道储存桶的名字,其次就是Access key ID和Secret access key package handl ...
- Hive sampling 语法之TABLESAMPLE用法理解
官网关于LanguageManual Sampling的教程,部分截图如下,这里主要分享对TABLESAMPLE(BUCKET 3 OUT OF 16 ON id)子句的理解 官网中假设创建表时设 ...
- 简明 Java 错误处理机制
用个最简单的例子来展示为什么需要错误处理,以及怎么用. import java.io.FileInputStream; import java.io.FileNotFoundException; pu ...
- vs2017专业版和企业版的密钥
Enterprise: NJVYC-BMHX2-G77MM-4XJMR-6Q8QF Professional: KBJFW-NXHK6-W4WJM-CRMQB-G3CDH
- C# vb .net图像合成-合成椭圆
在.net中,如何简单快捷地实现图像合成呢,比如合成文字,合成艺术字,多张图片叠加合成等等?答案是调用SharpImage!专业图像特效滤镜和合成类库.下面开始演示关键代码,您也可以在文末下载全部源码 ...
- maven项目整合SSM配置log4j, 实现控制台打印SQL语句
在原有项目正常启动的情况下, 实现在控制台打印mapper包下SQL语句. 1.在pom.xml配置文件中添加两个依赖(缺一不可) <!--日志包--> <dependency> ...
- Mybatis源码解析(二) —— 加载 Configuration
Mybatis源码解析(二) -- 加载 Configuration 正如上文所看到的 Configuration 对象保存了所有Mybatis的配置信息,也就是说mybatis-config. ...
- Linux命令:scp
1.find命令: scp [参数] [原路径] [目标路径] 2.用法: scp是 secure copy的缩写, scp是linux系统下基于ssh登陆进行安全的远程文件拷贝命令.linux的sc ...