[BZOJ4182]Shopping (点分治+树上多重背包+单调队列优化)
[BZOJ4182]Shopping (点分治+树上多重背包+单调队列优化)
题面
马上就是小苗的生日了,为了给小苗准备礼物,小葱兴冲冲地来到了商店街。商店街有n个商店,并且它们之间的道路构成了一颗树的形状。
第i个商店只卖第i种物品,小苗对于这种物品的喜爱度是wi,物品的价格为ci,物品的库存是di。但是商店街有一项奇怪的规定:如果在商店u,v买了东西,并且有一个商店w在u到v的路径上,那么必须要在商店w买东西。小葱身上有m元钱,他想要尽量让小苗开心,所以他希望最大化小苗对买到物品的喜爱度之和。这种小问题对于小葱来说当然不在话下,但是他的身边没有电脑,于是他打电话给同为OI选手的你,你能帮帮他吗?
分析
由题意,如果在u,v买了东西,那u到v的路径上的所有点一定都得买。因此最后购买东西的点一定是一个联通块。显然这个联通块一定包含某个子树的重心,所以可以点分治。
当我们点分治到某个重心rt的时候,我们考虑对这个重心分治出来子树计算背包。显然子树中的一个节点被选,它的父亲也一定被选。那么我们就把问题转化成了一般的树形背包问题,注意结点x必须选。
那么就可以列出模板的dp方程。设\(dp[i][j]\)表示当前处理到dfs序为i的节点x,当前背包内物品价格不大于j,能得到的最大喜爱度。设x子树的下一个子树dfs序的开始位置为nex[x].x为dfs序i对应的节点
则\(dp[i][j]=max(dp[nex[x]][j])\)(不选子树x)
\(dp[i][j]=max(dp[i+1][j-k \times w[x]]+k*c[x])\)(在x号节点买东西,买k个)
我们倒序dp是因为这样\(dp[0][m]\)存储的就是包含重心rt,且价格不大于m的最大喜爱度。可以直接用来更新答案
第二个方程显然可以按照多重背包的方法单调队列优化,这里不再赘述
总时间复杂度\(O(nm\log n)\)
代码
//http://119.29.55.79/contest/88/problem/2
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 500
#define maxv 4000
using namespace std;
int T,n,m;
struct edge{
	int from;
	int to;
	int next;
}E[maxn*2+5];
int esz=0;
int head[maxn+5];
void add_edge(int u,int v){
	esz++;
	E[esz].from=u;
	E[esz].to=v;
	E[esz].next=head[u];
	head[u]=esz;
}
int totsz,root;
bool vis[maxn+5];
int sz[maxn+5];
int maxsz[maxn+5];
void get_root(int x,int fa){
	sz[x]=1;
	maxsz[x]=0;
	for(int i=head[x];i;i=E[i].next){
		int y=E[i].to;
		if(y!=fa&&!vis[y]){
			get_root(y,x);
			sz[x]+=sz[y];
			maxsz[x]=max(maxsz[x],sz[y]);
		}
	}
	maxsz[x]=max(maxsz[x],totsz-sz[x]);
	if(maxsz[x]<maxsz[root]) root=x;
}
int tim;
int dfn[maxn+5],hash_dfn[maxn+5];
int nex[maxn+5];
void get_dfn(int x,int fa){
	dfn[x]=tim++;
	hash_dfn[dfn[x]]=x;
	sz[x]=1;
	for(int i=head[x];i;i=E[i].next){
		int y=E[i].to;
		if(y!=fa&&!vis[y]){
			get_dfn(y,x);
			sz[x]+=sz[y];
		}
	}
	nex[x]=tim;
}
int q[maxv+5];
int w[maxn+5],c[maxn+5],d[maxn+5];
int dp[maxn+5][maxv+5];
int ans=0;
void solve(int rt){
	vis[rt]=1;
	tim=0;
	get_dfn(rt,0);
	//一定要选根节点
	//倒序循环,这样dp[0]就存储了选rt的答案
	memset(dp[tim],0,sizeof(dp[tim]));
	for(int i=tim-1;i>=0;i--){
		int x=hash_dfn[i];
		for(int j=0;j<=m;j++) dp[i][j]=dp[nex[x]][j];//不选
		for(int b=0;b<c[x];b++){//选,多重背包单调队列优化
			int head=1,tail=0;
			for(int a=0;a*c[x]+b<=m;a++){
				while(head<=tail&&a-q[head]>d[x]) head++;
				if(head<=tail) dp[i][a*c[x]+b]=max(dp[i][a*c[x]+b],dp[i+1][q[head]*c[x]+b]+(a-q[head])*w[x]);
				int now=dp[i+1][a*c[x]+b]-a*w[x];
				while(head<=tail&&now>=dp[i+1][q[tail]*c[x]+b]-q[tail]*w[x]) tail--;
				q[++tail]=a;
			}
		}
	}
	for(int i=1;i<=m;i++) ans=max(ans,dp[0][i]);
	for(int i=head[rt];i;i=E[i].next){
		int y=E[i].to;
		if(!vis[y]){
			root=0;
			maxsz[root]=sz[y];
			totsz=sz[y];
			get_root(y,rt);
			solve(root);
		}
	}
}
void ini(){
	esz=0;
	memset(head,0,sizeof(head));
	ans=0;
	memset(dp,0,sizeof(dp));
	memset(vis,0,sizeof(vis));
	memset(sz,0,sizeof(sz));
	memset(maxsz,0,sizeof(maxsz));
	memset(dfn,0,sizeof(dfn));
	memset(hash_dfn,0,sizeof(hash_dfn));
	memset(nex,0,sizeof(nex));
	memset(q,0,sizeof(q));
}
int main(){
//	freopen("1.in","r",stdin);
	int u,v;
	scanf("%d",&T);
	while(T--){
		ini();
		scanf("%d %d",&n,&m);
		for(int i=1;i<=n;i++) scanf("%d",&w[i]);
		for(int i=1;i<=n;i++) scanf("%d",&c[i]);
		for(int i=1;i<=n;i++) scanf("%d",&d[i]);
		for(int i=1;i<n;i++){
			scanf("%d %d",&u,&v);
			add_edge(u,v);
			add_edge(v,u);
		}
		root=0;
		maxsz[root]=n;
		totsz=n;
		get_root(1,0);
		solve(root);
		printf("%d\n",ans);
	}
}
												
											[BZOJ4182]Shopping (点分治+树上多重背包+单调队列优化)的更多相关文章
- BZOJ 4182 Shopping (点分治+树上多重背包)
		
题目大意:给你一颗树,你有$m$元钱,每个节点都有一种物品,价值为$w$,代价为$c$,有$d$个,如果在$u$和$v$两个城市都购买了至少一个物品,那么$u,v$路径上每个节点也都必须买至少一个物品 ...
 - 【POJ1276】Cash Machine(多重背包单调队列优化)
		
大神博客转载http://www.cppblog.com/MatoNo1/archive/2011/07/05/150231.aspx多重背包的单调队列初中就知道了但一直没(不会)写二进制优化初中就写 ...
 - Luogu 3423 [POI 2005]BAN-银行票据 (多重背包单调队列优化 + 方案打印)
		
题意: 给出 n 种纸币的面值以及数量,求最少使用多少张纸币能凑成 M 的面额. 细节: 好像是要输出方案,看来很是头疼啊. 分析: 多重背包,裸体??? 咳咳,好吧需要低调,状态就出来了: dp [ ...
 - poj1742 Coins(多重背包+单调队列优化)
		
/* 这题卡常数.... 二进制优化或者单调队列会被卡 必须+上个特判才能过QAQ 单调队列维护之前的钱数有几个能拼出来的 循环的时候以钱数为步长 如果队列超过c[i]就说明队头的不能再用了 拿出来 ...
 - hdu 2844 多重背包+单调队列优化
		
思路:把价值看做体积,而价值的大小还是其本身,那么只需判断1-m中的每个状态最大是否为自己,是就+1: #include<iostream> #include<algorithm&g ...
 - BZOJ.4182.Shopping(点分治/dsu on tree 树形依赖背包 多重背包 单调队列)
		
BZOJ 题目的限制即:给定一棵树,只能任选一个连通块然后做背包,且每个点上的物品至少取一个.求花费为\(m\)时最大价值. 令\(f[i][j]\)表示在点\(i\),已用体积为\(j\)的最大价值 ...
 - bzoj 1531 Bank notes 多重背包/单调队列
		
多重背包二进制优化终于写了一次,注意j的边界条件啊,疯狂RE(还是自己太菜了啊啊)最辣的辣鸡 #include<bits/stdc++.h> using namespace std; in ...
 - POJ 1742 Coins(多重背包, 单调队列)
		
Description People in Silverland use coins.They have coins of value A1,A2,A3...An Silverland dollar. ...
 - 多重背包 /// 单调队列DP oj1943
		
题目大意: em.... 就是多重背包 挑战340页的东西 ...自己的笔记总结总是比较乱的 重点:原始的状态转移方程中 更新第 i 种物品时 重量%w[i] 的值不同 则它们之间是相互独立的: 1- ...
 
随机推荐
- 原生JS 将时间转换成几秒前,几分钟前…常用于评论回复功能
			
//格式化时间 留备用~ function getDateDiff(dateStr) { var publishTime = dateStr / 1000, d_seconds, d_minutes, ...
 - mysql BETWEEN操作符 语法
			
mysql BETWEEN操作符 语法 作用:选取介于两个值之间的数据范围.这些值可以是数值.文本或者日期.大理石平台 语法:SELECT column_name(s) FROM table_name ...
 - mysql SELECT语句 语法
			
mysql SELECT语句 语法,苏州大理石方箱 作用:用于从表中选取数据.结果被存储在一个结果表中(称为结果集). 语法:SELECT 列名称 FROM 表名称 以及 SELECT * FROM ...
 - Java数据结构之排序---冒泡排序
			
冒泡排序的基本思想: 通过对待排序序列从前到后(从下标小的元素开始),依次比较相邻位置的元素的值,若发现与给定的次序冲突,则交换位置(假设数值大的数放在序列的后面),使数值较大的元素逐渐从前移动到后部 ...
 - 基础:高通bring up camera【转】
			
本文转载自:http://blog.csdn.net/liwei16611/article/details/53056710 bring UP 需要做的工作 1.kernelspace senso ...
 - page size
			
https://dev.mysql.com/doc/refman/5.7/en/glossary.html#glos_page_size https://dev.mysql.com/doc/refma ...
 - LongAdder 源码分析
			
LongAdder LongAdder 能解决什么问题?什么时候使用 LongAdder? 1)LongAdder 内部包含一个基础值[base]和一个单元[Cell]数组. 没有竞争的情况下,要累加 ...
 - AtomicReference 源码分析
			
AtomicReference AtomicReference 能解决什么问题?什么时候使用 AtomicReference? 1)AtomicReference 可以原子更新引用对象. 2)comp ...
 - 自动化生成 Openstack 新项目开发框架
			
目录 目录 前言 环境 openstack-project-generator 前言 Openstack Developer 应该都知道, 开发一个 Openstack 的新项目并不是一个从 0 到 ...
 - 阶段1 语言基础+高级_1-3-Java语言高级_05-异常与多线程_第2节 线程实现方式_1_并发与并行
			
并发,相当于 一个人吃两个馒头,吃一口这个再吃一口另外一个.这里是cpu一会执行任务1,一会又执行任务2 并行,相当于两个人 吃两个馒头,各自吃各自的,这样速度就会快