Bajtman i Okrągły Robin
Bajtman i Okrągły Robin
题目描述
你是一个保安,你发现有n个强盗,其中第i个强盗会在[a[i],a[i]+1],[a[i]+1,a[i]+2],...,[b[i]-1,b[i]]这么多段长度为1时间进行抢劫,并计划抢走c[i]元。你在一个强盗任何一个作案的时间阻止他,你就可以挽回这个强盗带来的损失。显然,对于每个强盗,你只能挽回一次损失。不幸的是,你在每一段长度为1的时间内最多只能制止一个强盗,那么你最多可以挽回多少损失呢?
输入
第一行包含一个正整数n(1<=n<=5000),表示强盗的个数。
接下来n行,每行包含三个正整数a[i],b[i],c[i](1<=a[i]<b[i]<=5000,1<=c[i]<=10000),依次描述每一个强盗。
输出
输出一个整数,即可以挽回的损失的最大值。
样例输入
4
1 4 40
2 4 10
2 3 30
1 3 20
样例输出
90
提示
注意题面中的时间段跟时间点是不一样的。
solution
考虑网络流
把时间段建成点,强盗也建成点,源点向时间段连边(1,0),强盗向汇点连边(1,0),时间段向强盗连边(1,-C[i]),
然后费用流即可。
然而边最多有5000*5000条,会T。
用线段树优化网络流建图:把时间段开成线段树每个,强盗连对应的节点。
子节点向父节点连边,S向叶子连边
由于一段区间对应线段树上不超过logn段区间,所以边数不超过nlogn
然后就可以了
吐槽数据,一直T
居然卡zkw
优化:
1.线段树不建满
2.把边反着建
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
#define maxn 30005
#define inf 1e9
using namespace std;
int n,co,S,T,tot=1,head[maxn],li,ri,cost;
int d[maxn],flag[maxn],ans,cur[maxn],pre[maxn];
queue<int>q;
struct node{
int id,v,nex,cap,w;
}e[200002];
struct no{
int l,r;
}tree[maxn*4];
int ll[5022],rr[5022],c[5022];
void lj(int t1,int t2,int t3,int t4){
tot++;e[tot].id=t1;e[tot].v=t2;e[tot].cap=t3;e[tot].w=t4;e[tot].nex=head[t1];head[t1]=tot;
}
void build(int k,int L,int R){
tree[k].l=L,tree[k].r=R;
if(L==R){
lj(S,k,1,0);lj(k,S,0,0);
co=max(co,k);
return;
}
int mid=L+R>>1;
build(k*2,L,mid);build(k*2+1,mid+1,R);
lj(k*2,k,mid-L+1,0);lj(k,k*2,0,0);
lj(k*2+1,k,R-mid,0);lj(k,k*2+1,0,0);
}
void jia(int k,int pl){
if(tree[k].l>=li&&tree[k].r<=ri){
lj(k,pl,1,-cost);lj(pl,k,0,cost);
return;}
int mid=tree[k].l+tree[k].r>>1;
if(li<=mid)jia(k*2,pl);
if(ri>mid)jia(k*2+1,pl);
}
bool SPFA()
{
for(int i=1;i<=T;i++)flag[i]=0,d[i]=inf;
d[S]=0;flag[S]=1;q.push(S);
while(!q.empty())
{
int x=q.front();q.pop();
for(int i=head[x];i;i=e[i].nex){
if(e[i].cap>0&&d[e[i].v]>d[x]+e[i].w){
d[e[i].v]=d[x]+e[i].w;pre[e[i].v]=i;
if(!flag[e[i].v])
{
flag[e[i].v]=1;q.push(e[i].v);
}
}
}
flag[x]=0;
}
if(d[T]==inf)return 0;
int u=pre[T],mi=inf;
while(1)
{
mi=min(e[u].cap,mi);
if(e[u].id==S)break;
u=pre[e[u].id];
}
u=pre[T];
while(1)
{
e[u].cap-=mi;e[u^1].cap+=mi;
if(e[u].id==S)break;
u=pre[e[u].id];
}
ans+=d[T]*mi;
return 1;
}
int main(){
cin>>n;
S=30000,T=30001;
int Max=0;
for(int i=1;i<=n;i++){
scanf("%d%d%d",&ll[i],&rr[i],&c[i]);
rr[i]--;Max=max(Max,rr[i]);
}
build(1,1,Max);
for(int i=1;i<=n;i++){
li=ll[i];ri=rr[i];cost=c[i];
jia(1,co+i);
lj(co+i,T,1,0);lj(T,co+i,0,0);
}
while(SPFA());
cout<<-ans<<endl;
return 0;
}
Bajtman i Okrągły Robin的更多相关文章
- BZOJ 4276: [ONTAK2015]Bajtman i Okrągły Robin [线段树优化建边]
4276: [ONTAK2015]Bajtman i Okrągły Robin 题意:\(n \le 5000\)个区间\(l,r\le 5000\),每个区间可以选一个点得到val[i]的价值,每 ...
- [ONTAK2015]Bajtman i Okrągły Robin
bzoj 4276: [ONTAK2015]Bajtman i Okrągły Robin Time Limit: 40 Sec Memory Limit: 256 MB Description 有 ...
- 4276: [ONTAK2015]Bajtman i Okrągły Robin
4276: [ONTAK2015]Bajtman i Okrągły Robin Time Limit: 40 Sec Memory Limit: 256 MBSubmit: 345 Solved ...
- 【BZOJ4276】[ONTAK2015]Bajtman i Okrągły Robin 线段树优化建图+费用流
[BZOJ4276][ONTAK2015]Bajtman i Okrągły Robin Description 有n个强盗,其中第i个强盗会在[a[i],a[i]+1],[a[i]+1,a[i]+2 ...
- BZOJ4276 : [ONTAK2015]Bajtman i Okrągły Robin
建立线段树, S向每个叶子连边,容量1,费用0. 孩子向父亲连边,容量inf,费用0. 每个强盗向T连边,容量1,费用为c[i]. 对应区间内的点向每个强盗,容量1,费用0. 求最大费用流即可. #i ...
- BZOJ 4276: [ONTAK2015]Bajtman i Okrągły Robin
最大权值匹配,贪心匈牙利即可. 检查一些人是否能被全部抓住可以采用左端点排序,右端点优先队列处理. By:大奕哥 #include<bits/stdc++.h> using namespa ...
- BZOJ 4276 [ONTAK2015]Bajtman i Okrągły Robin 费用流+线段树优化建图
Description 有n个强盗,其中第i个强盗会在[a[i],a[i]+1],[a[i]+1,a[i]+2],...,[b[i]-1,b[i]]这么多段长度为1时间中选出一个时间进行抢劫,并计划抢 ...
- bzoj 4276: [ONTAK2015]Bajtman i Okrągły Robin【线段树+最大费用最大流】
--因为T点忘记还要+n所以选小了所以WA了一次 注意!题目中所给的时间是一边闭一边开的区间,所以读进来之后先l++(或者r--也行) 线段树优化建图,很神.(我记得还有个主席树优化建树的?)首先考虑 ...
- BZOJ_4276_[ONTAK2015]Bajtman i Okrągły Robin_线段树优化建图+最大费用最大流
BZOJ_4276_[ONTAK2015]Bajtman i Okrągły Robin_线段树优化建图+最大费用最大流 Description 有n个强盗,其中第i个强盗会在[a[i],a[i]+1 ...
随机推荐
- python_28_dictionary补充
#update:合并两个字典,如果有交叉就覆盖更新,没有交叉的就创建 info={ 'stu1101':'Liu Guannan', 'stu1102':'Wang Ruipu', 'stu1103' ...
- Websocket教程SpringBoot+Maven整合(详情)
1.大话websocket及课程介绍 简介: websocket介绍.使用场景分享.学习课程需要什么基础 笔记: websocket介绍: WebSocket协议是基于TCP的一种新的网络协议.它实现 ...
- Fight Against Traffic -简单dijkstra算法使用
题目链接 http://codeforces.com/contest/954/problem/D 题目大意 n m s t 分别为点的个数, 边的个数,以及两个特殊的点 要求s与t间的距离在新增一条边 ...
- 开发监测keepalived裂脑的脚本
检测思路:在备节点上执行脚本,如果可以ping通主节点并且备节点有VIP就报警,让人员介入检查是否裂脑. 在LB02备节点上开发脚本并执行: [root@lb02 ~]# cat /server/sc ...
- Python_深浅拷贝
深浅拷贝 ‘copy’和'='的区别:copy会开辟一个新的空间,而‘=’不会. 浅copy只会copy第一层,再里边的就进行共享了. 需要记住的是copy之后记住的是内存寻址地址,而浅copy时如果 ...
- 721. Accounts Merge
https://leetcode.com/problems/accounts-merge/description/ class UnionFound { public: unordered_map&l ...
- UVa 1455 Kingdom 线段树 并查集
题意: 平面上有\(n\)个点,有一种操作和一种查询: \(road \, A \, B\):在\(a\),\(b\)两点之间加一条边 \(line C\):询问直线\(y=C\)经过的连通分量的个数 ...
- HDU 4919 Exclusive or 数学
题意: 定义 \[f(n)=\sum\limits_{i=1}^{n-1}(i\oplus (n-i))\] 求\(f(n),n \leq 10^{500}\) 分析: 这个数列对应OEIS的A006 ...
- P2920 [USACO08NOV]时间管理Time Management
P2920 [USACO08NOV]时间管理Time Management 题目描述 Ever the maturing businessman, Farmer John realizes that ...
- 求 1 到 n 的所有数的约数和
求 1 到 n 的所有数的约数和 暴力方法就是枚举每个数,算出他的约数和即可,这样有点慢. 另一种思路,枚举约数,判断他是谁的约数,并记录(即他的倍数有多少个),在乘以他自己. n/i求的是n以内,i ...