旅行商(sale)
旅行商(sale)
题目描述
camp国有n座城市,由1,2,…,n编号。城市由n–1条双向道路相连。任意两个城市之间存在唯一的道路连通。有m个旅行商,第i个旅行商会从城市ai旅行到城市bi,贩卖ci件商品。已知第i个城市的居民最多购买wi件商品,bobo想知道旅行商们能够卖出商品数量的最大值。
输入
输入文件sale.in。
第一行,包含2个整数n和m。
第二行,包含n个整数w1,w2,…,wn。
接下来n–1行中的第i行包含2个整数ui,vi代表城市ui和vi之间有一条道路。
最后的m行中的第i行包含3个整数ai,bi,ci。
输出
输出文件sale.out。
1个整数,代表卖出商品数量的最大值。
样例输入
4 2
0 1 2 2
1 4
2 4
3 4
1 2 2
1 3 3
样例输出
5
提示
【数据范围和约定】
|
数据点 |
n,m的范围 |
其他 |
|
1 |
≤500 |
1≤ui,vi,ai,bi≤n 0≤wi,ci≤105 |
|
2 |
||
|
3 |
||
|
4 |
≤5,000 |
|
|
5 |
||
|
6 |
||
|
7 |
≤2×104 |
|
|
8 |
||
|
9 |
||
|
10 |
solution
树剖,转化为区间问题
先暴力建图
每个旅行商向他要去的所有城市连边,发现边数n^2
考虑用线段树优化建图,城市开成线段树,旅行商向节点连边
然后就可以啦
好裸的题
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
#include<queue>
#define maxn 20005
#define N 400005
#define inf 1e9
using namespace std;
int n,m,w[maxn],dfn[maxn],sc,son[maxn],fa[maxn],top[maxn],size[maxn];
int deep[maxn],t1,t2,t3,tot=1,a,b,c,dy[maxn],li,ri,cnt,S,T,ans;
int head[N],cur[N],flag[N],d[N];
queue<int>q;
vector<int>G[maxn];
struct node{
int v,nex,cap;
}e[N*20];
struct no{
int l,r,x;
}tree[maxn*4];
void lj(int t1,int t2,int t3){
e[++tot].v=t2;e[tot].cap=t3;e[tot].nex=head[t1];head[t1]=tot;
e[++tot].v=t1;e[tot].cap=0;e[tot].nex=head[t2];head[t2]=tot;
}
void dfs1(int k,int fath){
fa[k]=fath;deep[k]=deep[fath]+1;
int sz=0,gp=-1;
for(int i=0;i<G[k].size();i++){
int v=G[k][i];
if(v!=fath){
dfs1(v,k);
sz+=size[v];
if(gp==-1)gp=v;
else if(size[v]>size[gp])gp=v;
}
}
size[k]=sz+1;son[k]=gp;
}
void dfs2(int k){
dfn[k]=++sc;dy[sc]=k;
if(son[k]!=-1)top[son[k]]=top[k],dfs2(son[k]);
for(int i=0;i<G[k].size();i++){
int v=G[k][i];
if(v!=fa[k]&&v!=son[k]){
top[v]=v;dfs2(v);
}
}
}
void wh(int k){
tree[k].x=tree[k*2].x+tree[k*2+1].x;
}
void build(int k,int L,int R){
tree[k].l=L,tree[k].r=R;
if(L==R){
tree[k].x=w[dy[L]];lj(k,T,w[dy[L]]);
cnt=max(cnt,k);
return;
}
int mid=L+R>>1;
build(k*2,L,mid);build(k*2+1,mid+1,R);
lj(k,k*2,tree[k*2].x);lj(k,k*2+1,tree[k*2+1].x);
wh(k);
}
void fsy(int k,int id){
if(tree[k].l>=li&&tree[k].r<=ri){
lj(id,k,inf);return;
}
int mid=tree[k].l+tree[k].r>>1;
if(li<=mid)fsy(k*2,id);
if(ri>mid)fsy(k*2+1,id);
}
bool BFS(){
for(int i=1;i<=T;i++)d[i]=inf;
d[S]=0;q.push(S);
while(!q.empty()){
int x=q.front();q.pop();
cur[x]=head[x];
for(int i=head[x];i;i=e[i].nex){
if(d[e[i].v]>d[x]+1&&e[i].cap>0){
d[e[i].v]=d[x]+1;
if(!flag[e[i].v]){
flag[e[i].v]=1;q.push(e[i].v);
}
}
}
flag[x]=0;
}
return d[T]!=inf;
}
int lian(int k,int a){
if(k==T||!a)return a;
int f,flow=0;
for(int &i=cur[k];i;i=e[i].nex){
if(d[e[i].v]==d[k]+1&&(f=lian(e[i].v,min(e[i].cap,a)))>0){
e[i].cap-=f;e[i^1].cap+=f;
a-=f;flow+=f;
if(!a)break;
}
}
return flow;
}
int main()
{
cin>>n>>m;S=400001,T=S+1;
for(int i=1;i<=n;i++)scanf("%d",&w[i]);
for(int i=1;i<n;i++){
scanf("%d%d",&t1,&t2);
G[t1].push_back(t2);G[t2].push_back(t1);
}
dfs1(1,0);top[1]=1;dfs2(1);
build(1,1,n);
for(int i=1;i<=m;i++){
scanf("%d%d%d",&a,&b,&c);
int t1=top[a],t2=top[b];
while(t1!=t2){
if(deep[t1]<deep[t2])swap(t1,t2),swap(a,b);
li=dfn[t1],ri=dfn[a];
fsy(1,cnt+i);
a=fa[t1];t1=top[a];
}
if(deep[a]<deep[b])swap(a,b);
li=dfn[b],ri=dfn[a];
fsy(1,cnt+i);
lj(S,cnt+i,c);
}
while(BFS())ans+=lian(S,inf);
cout<<ans<<endl;
return 0;
}
旅行商(sale)的更多相关文章
- 【C#代码实战】群蚁算法理论与实践全攻略——旅行商等路径优化问题的新方法
若干年前读研的时候,学院有一个教授,专门做群蚁算法的,很厉害,偶尔了解了一点点.感觉也是生物智能的一个体现,和遗传算法.神经网络有异曲同工之妙.只不过当时没有实际需求学习,所以没去研究.最近有一个这样 ...
- 洛谷P1782 旅行商的背包[多重背包]
题目描述 小S坚信任何问题都可以在多项式时间内解决,于是他准备亲自去当一回旅行商.在出发之前,他购进了一些物品.这些物品共有n种,第i种体积为Vi,价值为Wi,共有Di件.他的背包体积是C.怎样装才能 ...
- 2016全国研究生数学建模A题多无人机协同任务规划——基于分布式协同多旅行商MTSP遗传算法
MTSP问题是指:有Ⅳ个城市,要求旅行商到达每个城市各一次,且仅一次,并[旦 1到起点,且要求旅行路线最短.而多旅行商问题M个旅行商从同一个城市(或多个城市)出发.分羽走一条旅路线,且总路程缀短.有关 ...
- [vijos P1014] 旅行商简化版
昨天早上上课讲旅行商问题,有点难,这周抽空把3^n的算法码码看.不过这个简化版已经够折腾人了. 其一不看解析不知道这是双进程动态规划,不过我看的解析停留在f[i,j]表示第一个人走到i.第二个人走到j ...
- vijosP1014 旅行商简化版
vijosP1014 旅行商简化版 链接:https://vijos.org/p/1014 [思路] 双线DP. 设ab,ab同时走.用d[i][j]表示ab所处结点i.j,且定义i>j,则有转 ...
- 洛谷【P1523】旅行商的背包(算法导论 15-1) 题解
P1523 旅行商简化版 题目背景 欧几里德旅行商\((Euclidean Traveling Salesman)\)问题也就是货郎担问题一直是困扰全世界数学家.计算机学家的著名问题.现有的算法都没有 ...
- Bzoj3352 [ioi2009]旅行商
Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 89 Solved: 36 Description 旅行商认定如何优化旅行路线是一个非常棘手的计算问题 ...
- hdu 4281 Judges' response(多旅行商&DP)
Judges' response Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- 洛谷P1523 旅行商简化版(DP)
题目: P1523 旅行商简化版 解析 可以看做是两个人同时从西往东走,经过不一样的点,走到最东头的方案数 设\(f[i][j]\)表示一个人走到i,一个人走到j的最短距离(\(i<j\)) 第 ...
随机推荐
- nginx入门学习步骤(linux)
一.nginx下载(nginx-1.9.9) http://nginx.org/download/ 二.解压到指定文件夹 tar -zxvf 解压缩文件 三.设置配置信息 在nignx解压文件夹内执行 ...
- python-的多线程处理
书到用时方恨少,这句话在软件杯真的深深体会到了.但是对自己对于代码的领会能力还是有自信的,在做项目的时候用到了多线程的处理,开始都不知道该怎么去百度搜索关键词
- java基础面试题:java中有没有goto? 在JAVA中如何跳出当前的多重嵌套循环?
goto是Java的保留关键字,但是从来没有直接使用到goto,goto的跳转作用在Java中受到了限制,只有在特定场合下才能使用,如while for等循环中用continue.break或结合标签 ...
- 去除select下拉框默认样式
去除select下拉框默认样式 select { /*Chrome和Firefox里面的边框是不一样的,所以复写了一下*/ border: solid 1px #; /*很关键:将默认的select选 ...
- Vue 恢复初始值的快速方法
vue 中经常定义很多data ,在用户进行一些操作后,需要讲data中的某个对象定义为初始值 例如 form: { title: '', describe: '', inspectionCatego ...
- 4396: [Usaco2015 dec]High Card Wins
Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 275 Solved: 175[Submit][Status][Discuss] Descriptio ...
- 三、Shell 传递参数
Shell 传递参数 我们可以在执行 Shell 脚本时,向脚本传递参数,脚本内获取参数的格式为:$n.n 代表一个数字,1 为执行脚本的第一个参数,2 为执行脚本的第二个参数,以此类推…… 实例 以 ...
- Python_三级目录
程序要求: 1. 使用字典存储 1. 可以一层一层的进入到所有层2. 可以在每层返回上一层3. 可以在任意层退出 三级目录写了两个版本,第一个版本是刚看完字典写出来的,代码很多冗余,很多重复. men ...
- 第八篇:ORM框架SQLAlchemy 了解知识
一 介绍 SQLAlchemy是Python编程语言下的一款ORM框架,该框架建立在数据库API之上,使用关系对象映射进行数据库操作,简言之便是:将对象转换成SQL,然后使用数据API执行SQL并获取 ...
- 绘制弧形:imagearc() 说明:三点钟的位置是起点(0度
<?php //1. 绘制图像资源(创建一个画布) $image = imagecreatetruecolor(500, 300); //2. 先分配一个绿色 $green = imagecol ...