4182: Shopping


Time Limit: 30 Sec  Memory Limit: 128 MB
Submit: 374  Solved: 130
[Submit][Status][Discuss]

Description


马上就是小苗的生日了,为了给小苗准备礼物,小葱兴冲冲地来到了商店街。商店街有n个商店,并且它们之间的道路构成了一颗树的形状。

第i个商店只卖第i种物品,小苗对于这种物品的喜爱度是wi,物品的价格为ci,物品的库存是di。但是商店街有一项奇怪的规定:如果在商店u,v买了东西,并且有一个商店w在u到v的路径上,那么必须要在商店w买东西。小葱身上有m元钱,他想要尽量让小苗开心,所以他希望最大化小苗对买
到物品的喜爱度之和。这种小问题对于小葱来说当然不在话下,但是他的身边没有电脑,于是他打电话给同为OI选手的你,你能帮帮他吗?
 

Input


输入第一行一个正整数T,表示测试数据组数。

对于每组数据,
第一行两个正整数n;m;
第二行n个非负整数w1,w2...wn;
第三行n个正整数c1,c2...cn;
第四行n个正整数d1,d2...dn;
接下来n-1行每行两个正整数u;v表示u和v之间有一条道路
 

Output


输出共T 行,每行一个整数,表示最大的喜爱度之和。
 

Sample Input



Sample Output



HINT


N<=500,M<=4000,T<=5,Wi<=4000,Di<=100

分析:


题意:两个点选了,它路径上的点必须选。求树上一个联通块的多重背包,权值最大。

题解:
 先用点分治假设重心必选,然后dfs它子树,这样每个点会做背包的次数降低到log次。
 
 然后dfs子树时列出dfs序,然后转移方程:
 

对于第二步,多重背包优化可以考虑二进制拆分总复杂度为O(Tnmlognlogm)。

也可以使用单调队列优化总复杂度O(Tnmlogn)

下面是对单调队列优化的图解:

嗯。。没了。

AC代码:


# include <cstdio>
# include <cstring>
# include <iostream>
# include <algorithm>
using namespace std;
const int N = 4e3 + ;
const int M = 5e3 + ;
int mx[N],w[N],v[N],c[N],n,m,root,head[N],dt,sz[N],sum,id[N],ed[N],ans;
int f[N][M];
bool vis[N];
struct Edge{
int to,nex;
}edge[N << ];
void AddEdge(int u,int v)
{
edge[++dt] = (Edge){v,head[u]};
head[u] = dt;
}
void find(int u,int pre)
{
mx[u] = ;sz[u] = ;
for(int i = head[u];i;i = edge[i].nex)
{
if(vis[edge[i].to] || edge[i].to == pre)continue;
find(edge[i].to,u);
sz[u] += sz[edge[i].to];
mx[u] = max(mx[u],sz[edge[i].to]);
}
mx[u] = max(mx[u],sum - sz[u]);
if(mx[u] < mx[root])root = u;
}
void dfs(int u,int pre)
{
sz[u] = ;id[++dt] = u;
for(int i = head[u];i;i = edge[i].nex)
{
if(vis[edge[i].to] || edge[i].to == pre)continue;
dfs(edge[i].to,u);
sz[u] += sz[edge[i].to];
}
ed[u] = dt;
}
int Q1[M],Q2[M];
void calc(int *g,int x)
{
int h1,h2,t1,t2,cnt,t;
for(int j = ;j < v[x];j++)
{
h1 = t1 = h2 = t2 = cnt = ;
for(int k = j;k <= m;k += v[x])
{
if(t1 - h1 == c[x] + ){
if(Q2[h2 + ] == Q1[h1 + ])++h2;
++h1;
}
t = g[k] - cnt * w[x];
Q1[++t1] = t;
while(h2 < t2 && Q2[t2] < t)--t2;
Q2[++t2] = t;
g[k] = Q2[h2 + ] + cnt * w[x];
++cnt;
}
}
}
void solve()
{
for(int i = ;i <= dt + ;i++)
for(int j = ;j <= m;j++)
f[i][j] = ;
int x,t;
for(int i = dt;i >= ;i--)
{
x = id[i];
for(int j = m;j >= v[x];j--)f[i][j] = f[i + ][j - v[x]] + w[x];
calc(f[i],x);
for(int j = m;j >= ;j--)f[i][j] = max(f[i][j],f[ed[x] + ][j]);
}
ans = max(ans,f[][m]);
}
void dfs(int u)
{
vis[u] = true;dt = ;
dfs(u,-);
solve();
for(int i = head[u];i;i = edge[i].nex)
{
if(vis[edge[i].to])continue;
root = ;sum = sz[edge[i].to];
if(sum > sz[u])sum = sum - sz[u];
find(edge[i].to,u);
dfs(root);
}
}
int main()
{
mx[] = N;
int Case;scanf("%d",&Case);
while(Case--)
{
scanf("%d %d",&n,&m);int x,y;
memset(vis,false,sizeof vis);
memset(head,,sizeof head);dt = ans = ;
for(int i = ;i <= n;i++)scanf("%d",&w[i]);
for(int i = ;i <= n;i++)scanf("%d",&v[i]);
for(int i = ;i <= n;i++)scanf("%d",&c[i]),c[i]--;
for(int i = ;i < n;i++)scanf("%d %d",&x,&y),AddEdge(x,y),AddEdge(y,x);
root = ;sum = n;find(,-);
dfs(root);
printf("%d\n",ans);
}
}
 

[Bzoj4182]Shopping(点分治)(树上背包)(单调队列优化多重背包)的更多相关文章

  1. POJ 1742 (单调队列优化多重背包+混合背包)

    (点击此处查看原题) 题意分析 给你n种不同价值的硬币,价值为val[1],val[2]...val[n],每种价值的硬币有num[1],num[2]...num[n]个,问使用这n种硬币可以凑齐[1 ...

  2. bzoj4182 Shopping 点分治+单调队列优化多重背包

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4182 题解 有一个很直观的想法是设 \(dp[x][i]\) 表示在以 \(x\) 为根的子树 ...

  3. POJ 1276 Cash Machine(单调队列优化多重背包)

    Cash Machine Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 38986   Accepted: 14186 De ...

  4. HDU 2191 - 单调队列优化多重背包

    题目: 传送门呀传送门~ Problem Description 急!灾区的食物依然短缺! 为了挽救灾区同胞的生命,心系灾区同胞的你准备自己采购一些粮食支援灾区,现在假设你一共有资金n元,而市场有m种 ...

  5. [BZOJ4182]Shopping (点分治+树上多重背包+单调队列优化)

    [BZOJ4182]Shopping (点分治+树上多重背包+单调队列优化) 题面 马上就是小苗的生日了,为了给小苗准备礼物,小葱兴冲冲地来到了商店街.商店街有n个商店,并且它们之间的道路构成了一颗树 ...

  6. [小明打联盟][斜率/单调队列 优化dp][背包]

    链接:https://ac.nowcoder.com/acm/problem/14553来源:牛客网 题目描述 小明很喜欢打游戏,现在已知一个新英雄即将推出,他同样拥有四个技能,其中三个小技能的释放时 ...

  7. 【POJ1276】Cash Machine(多重背包单调队列优化)

    大神博客转载http://www.cppblog.com/MatoNo1/archive/2011/07/05/150231.aspx多重背包的单调队列初中就知道了但一直没(不会)写二进制优化初中就写 ...

  8. Luogu 3423 [POI 2005]BAN-银行票据 (多重背包单调队列优化 + 方案打印)

    题意: 给出 n 种纸币的面值以及数量,求最少使用多少张纸币能凑成 M 的面额. 细节: 好像是要输出方案,看来很是头疼啊. 分析: 多重背包,裸体??? 咳咳,好吧需要低调,状态就出来了: dp [ ...

  9. poj1742 Coins(多重背包+单调队列优化)

    /* 这题卡常数.... 二进制优化或者单调队列会被卡 必须+上个特判才能过QAQ 单调队列维护之前的钱数有几个能拼出来的 循环的时候以钱数为步长 如果队列超过c[i]就说明队头的不能再用了 拿出来 ...

随机推荐

  1. JDBC基础-setFetchSize方法

    在Statement和ResultSet接口中都有setFetchSize方法 void setFetchSize(int rows) throws SQLException 查看API文档 Stat ...

  2. axios的简单封装及在组件内使用

    /**第一步 * 配置编译环境和线上环境之间的切换 * baseUrl: 域名地址 * routerMode: 路由模式 * imgBaseUrl: 图片所在域名地址 * */ let Host = ...

  3. Bug的分类和管理流程

    1.按照严重程度划分 定义:是指Bug对软件质量的破坏程度,即BUG的存在将对软件的功能和性能产生怎样的影响 分类:系统崩溃.严重.一般.次要.建议 2.按优先级划分 定义:表示处理和修正软件缺陷的现 ...

  4. Tunnelier使用说明

    Tunnelier与MyEnTunnel类似,但是功能更加强大.MyEnTunnel小巧易用,如何使用MyEnTunnel可以参考 MyEnTunnel使用说明 这里列下Tunnelier的优点: 1 ...

  5. ftp上传文件、删除文件、下载文件的操作

    FavFTPUtil.Java 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 ...

  6. 用户管理命令--passwd,usermod,userdel

    用户修改密码命令--passwd 当修改用户的密码时,也要分普通用户和超级用户两种情况 普通用户:修改密码前需要先输入当前密码,确认是否正确 密码设置不可以过于简单 超级用户:权利非常的大,可以设置任 ...

  7. cmd启动MySQL服务器发生错误

    Mysql net start mysql启动,提示发生系统错误 5 拒绝访问  原文:https://blog.csdn.net/angel_guoo/article/details/7919037 ...

  8. 5. COLUMNS

    5. COLUMNS 表COLUMNS提供表中列的信息. INFORMATION_SCHEMA Name SHOW Name Remarks TABLE_CATALOG TABLE_SCHEMA TA ...

  9. 编码&解码

    编码与解码首先,明确一点,计算机中存储的信息都是二进制的 编码/解码本质上是一种映射(对应关系):比如‘a’用ascii编码则是65,计算机中存储的就是00110101,但是显示的时候不能显示0011 ...

  10. MariaDB数据库(二)

    1. MariaDB数据类型 MariaDB数据类型可以分为数字,日期和时间以及字符串值. 使用数据类型的原则:够用就行,尽量使用范围小的,而不用大的. 1.1  常用的数据类型 整数:int,bit ...