线性规划||网络流(费用流):COGS 288. [NOI2008] 志愿者招募
[NOI2008] 志愿者招募
输入文件:employee.in
输出文件:employee.out
简单对比
时间限制:2 s
内存限制:512 MB
【问题描述】
申奥成功后,布布经过不懈努力,终于成为奥组委下属公司人力资源部门的主管。布布刚上任就遇到了一个难题:为即将启动的奥运新项目招募一批短期志愿者。经过估算,这个项目需要N 天才能完成,其中第i 天至少需要Ai 个人。
布布通过了解得知,一共有M 类志愿者可以招募。其中第i 类可以从第Si 天工作到第Ti 天,招募费用是每人Ci
元。新官上任三把火,为了出色地完成自己的工作,布布希望用尽量少的费用招募足够的志愿者,但这并不是他的特长!于是布布找到了你,希望你帮他设计一种最
优的招募方案。
【输入格式】
输入文件的第一行包含两个整数N, M,表示完成项目的天数和可以招募的志愿者的种类。
接下来的一行中包含N 个非负整数,表示每天至少需要的志愿者人数。
接下来的M 行中每行包含三个整数Si, Ti, Ci,含义如上文所述。为了方便起见,我们可以认为每类志愿者的数量都是无限多的。
【输出格式】
输入文件中仅包含一个整数,表示你所设计的最优方案的总费用。
【输入样例】
3 3
2 3 4
1 2 2
2 3 5
3 3 2
【输出样例】
14
【样例说明】
招募3 名第一类志愿者和4 名第三类志愿者。
【数据规模和约定】
30%的数据中,1 ≤ N, M ≤ 10,1 ≤ Ai ≤ 10;
100%的数据中,1 ≤ N ≤ 1000,1 ≤ M ≤ 10000,题目中其他所涉及的数据均
不超过2^31-1。
https://www.byvoid.com/zhs/blog/noi-2008-employee
#include <iostream>
#include <cstring>
#include <cstdio>
#include <queue>
using namespace std;
const int INF=;
const int maxn=;
const int maxm=;
int cnt=,fir[maxn],to[maxm],nxt[maxm],cap[maxm],val[maxm];
void addedge(int a,int b,int c,int v){
nxt[++cnt]=fir[a];to[cnt]=b;cap[cnt]=c;val[cnt]=v;fir[a]=cnt;
}
int N,M,S,T;
int dis[maxn],path[maxn],tot[maxn],vis[maxn]; queue<int>q;
int MCMF(){
int ret=;
while(true){
memset(dis,,sizeof(dis));dis[S]=;
q.push(S);vis[S]=;
while(!q.empty()){
int node=q.front();q.pop();vis[node]=;
for(int i=fir[node];i;i=nxt[i])
if(cap[i]&&dis[to[i]]>dis[node]+val[i]){
dis[to[i]]=dis[node]+val[i];
path[to[i]]=i;
if(!vis[to[i]]){
vis[to[i]]=;
q.push(to[i]);
}
}
}
if(dis[T]==)
break; int p=T,f=INF;
while(p!=S){
f=min(f,cap[path[p]]);
p=to[path[p]^];
}
ret+=dis[T]*f;p=T;
while(p!=S){
cap[path[p]]-=f;
cap[path[p]^]+=f;
p=to[path[p]^];
}
}
return ret;
} int main(){
freopen("employee.in","r",stdin);
freopen("employee.out","w",stdout);
scanf("%d%d",&N,&M);
S=;T=N+;
for(int i=;i<=N;i++)
scanf("%d",&tot[i]);
for(int i=,a,b,c;i<=M;i++){
scanf("%d%d%d",&a,&b,&c);
addedge(a,b+,INF,c);
addedge(b+,a,,-c);
}
for(int i=;i<=N+;i++){
int c=tot[i]-tot[i-];
if(c>){
addedge(S,i,c,);
addedge(i,S,,);
}
if(c<){
addedge(i,T,-c,);
addedge(T,i,,);
}
if(i>){
addedge(i,i-,INF,);
addedge(i-,i,,);
}
}
printf("%d\n",MCMF());
return ;
}
然后又用线性规划。
额,松弛型先天不足,数组开不了,无法通过,再学一下吧。
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cmath>
using namespace std;
const int maxr=;
const int maxc=; const double eps=1e-;
int n,m,nxt[maxc];
int N[maxr],B[maxr];
double a[maxr][maxr],v;
double b[maxr],c[maxr]; int SGN(double x){
return (x>eps)-(x<-eps);
} void Init(){
B[]=N[]=;v=0.0;
for(int i=;i<=n;i++)N[++N[]]=i;
for(int i=;i<=m;i++)B[++B[]]=i+n;
} void Pivot(int l,int e){
b[e]=b[l]/a[l][e];
a[e][l]=1.0/a[l][e];
for(int i=;i<=N[];i++)
if(N[i]!=e)a[e][N[i]]=a[l][N[i]]/a[l][e]; int pre=;
for(int i=;i<=N[];i++)
if(N[i]!=e&&SGN(a[e][N[i]])!=)
{nxt[pre]=i;pre=i;}
nxt[pre]=; for(int i=;i<=B[];i++)
if(B[i]!=l){
b[B[i]]-=a[B[i]][e]*b[e];
a[B[i]][l]=-a[B[i]][e]*a[e][l];
for(int j=nxt[];j;j=nxt[j])
if(N[j]!=e)a[B[i]][N[j]]-=a[B[i]][e]*a[e][N[j]];
} v+=c[e]*b[e];
c[l]=-c[e]*a[e][l];
for(int i=;i<=N[];i++)
if(N[i]!=e)
c[N[i]]-=c[e]*a[e][N[i]];
for(int i=;i<=N[];i++)if(N[i]==e)N[i]=l;
for(int i=;i<=B[];i++)if(B[i]==l)B[i]=e;
} void Simplex(){
while(true){
int e=maxr,l=maxr;
for(int i=;i<=N[];i++)
if(SGN(c[N[i]])>&&e>N[i])e=N[i]; if(e==maxr)break; double lam=-;
for(int i=;i<=B[];i++)
if(SGN(a[B[i]][e])>){
double tmp=b[B[i]]/a[B[i]][e];
if(lam==-||SGN(lam-tmp)>||SGN(lam-tmp)==&&l>B[i])
{lam=tmp;l=B[i];}
} Pivot(l,e);
}
} int main(){
#ifndef ONLINE_JUDGE
freopen("employee.in","r",stdin);
freopen("employee.out","w",stdout);
#endif
scanf("%d%d",&n,&m);
for(int i=,t;i<=n;i++){
scanf("%d",&t);
c[i]=t;
} for(int i=,l,r,t;i<=m;i++){
scanf("%d%d%d",&l,&r,&t);
for(int j=l;j<=r;j++)
a[i+n][j]=;
b[i+n]=t;
} Init();
Simplex(); printf("%d\n",(int)(v+0.5));
return ;
}
这个程序可以AC,但解法并不具有共性啊……
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
const int maxr=;
const int maxc=; int n,m,nxt[maxc];
int a[maxr][maxc]; void Pivot(int l,int e){
int pre=maxc-;
for(int i=;i<=n;i++)
if(a[l][i]!=){nxt[pre]=i;pre=i;}
nxt[pre]=-; for(int i=,t;i<=m;i++)
if(i!=l&&(t=a[i][e])){
a[i][e]=;
for(int j=nxt[maxc-];j!=-;j=nxt[j])
a[i][j]+=t*a[l][j];
}
} void Simplex(){
while(true){
int e=,l=;
for(int i=;i<=n;i++)
if(a[][i]>){e=i;break;}
if(e==)break;
for(int i=;i<=m;i++)
if(a[i][e]<&&(!l||a[l][]>a[i][]))
{l=i;} Pivot(l,e);
}
} int main(){
#ifndef ONLINE_JUDGE
freopen("employee.in","r",stdin);
freopen("employee.out","w",stdout);
#endif
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
scanf("%d",&a[][i]);
for(int i=,l,r,t;i<=m;i++){
scanf("%d%d%d",&l,&r,&t);
for(int j=l;j<=r;j++)
a[i][j]=-;
a[i][]=t;
}
Simplex();
printf("%d\n",a[][]);
return ;
}
这里的线性规划式子都是原式的对偶线性规划式。
线性规划||网络流(费用流):COGS 288. [NOI2008] 志愿者招募的更多相关文章
- 【费用流】BZOJ1061: [Noi2008]志愿者招募(这题超好)
1061: [Noi2008]志愿者招募 Time Limit: 20 Sec Memory Limit: 162 MBSubmit: 5291 Solved: 3173[Submit][Stat ...
- 线性规划费用流解法(Bzoj1061: [Noi2008]志愿者招募)
题面 传送门 Sol 线性规划费用流解法用与求解未知数为非负数的问题 这道题可以列出一堆形如 \(x[i]+x[j]+x[k]+...>=a[p]\) 的不等式 我们强行给每个式子减去一个东西, ...
- 【费用流】BZOJ1061[NOI2008]-志愿者招募
[题目大意] 一个项目需要n天完成,其中第i天至少需要Ai个人.共有m类人可以招募,其中第i类可以从第Si天做到第Ti天,每人的招募费用为Ci元.求最小招募费用. [思路] byvoid神犇的建图详解 ...
- [NOI2008] 志愿者招募[流量平衡]
288. [NOI2008] 志愿者招募 ★★★★ 输入文件:employee.in 输出文件:employee.out 简单对比时间限制:2 s 内存限制:512 MB [问题描述] ...
- 【bzoj1061】[NOI2008]志愿者招募 线性规划与费用流
题目描述 申奥成功后,布布经过不懈努力,终于成为奥组委下属公司人力资源部门的主管.布布刚上任就遇到了一个难题:为即将启动的奥运新项目招募一批短期志愿者.经过估算,这个项目需要N 天才能完成,其中第i ...
- 网络流解线性规划问题 BZOJ1061: [Noi2008]志愿者招募
线性规划定义: 在给定有限的资源和竞争约束情况下,很多问题都可以表述为最大化或最小化某个目标.如果可以把目标指定为某些变量的线性函数,而且如果可以将资源约束指定为这些变量的等式或不等式,则得到了一个线 ...
- 从[NOI2008志愿者招募]浅谈线性规划在网络流构图上的巧用
首先来看一下题..http://www.lydsy.com/JudgeOnline/problem.php?id=1061 1061: [Noi2008]志愿者招募 Description 申奥成功后 ...
- 【BZOJ 1061】 1061: [Noi2008]志愿者招募 (线性规划与网络流)**
1061: [Noi2008]志愿者招募 Description 申奥成功后,布布经过不懈努力,终于成为奥组委下属公司人力资源部门的主管.布布刚上任就遇到了一个难 题:为即将启动的奥运新项目招募一批短 ...
- 【费用流】NOI2008志愿者招募
1061: [Noi2008]志愿者招募 Time Limit: 20 Sec Memory Limit: 162 MBSubmit: 5171 Solved: 3089[Submit][Stat ...
随机推荐
- 使用PuTTY在Windows中向Linux上传文件
为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/SJQ. http://www.cnblogs.com/shijiaqi1066/p/3843207.html ...
- asp.net 的状态管理
状态管理 (state management) 在Web应用程序中,一向是很重要的课题,良好的状态管理可以帮助开发人员发展出具有状态持续能力的应用程序(像是工作流程型应用程序或是电子商务应用程序),但 ...
- hbase的存储体系
一.了解hbase的存储体系. hbase的存储体系核心的有Split机制,Flush机制和Compact机制. 1.split机制 每一个hbase的table表在刚刚开始的时候,只有一个regio ...
- Proxy 模式
在以下集中情况下可以用 Proxy模式解决问题: 1)创建开销大的对象时候,比如显示一幅大的图片,我们将这个创建的过程交给代理去完成,GoF 称之为虚代理(Virtual Proxy): 2)为网络上 ...
- [LeetCode OJ]-Climbing Stairs
You are climbing a stair case. It takes n steps to reach to the top. Each time you can either climb ...
- 记一个问题的AC
今天突然做一道LCT的染色问题的时候突然想到一个两个月前一道没有AC的题目. 链接 大意是,给一个长度为10^4的序列,最多有255个不同的数字,有最多10^5次方个询问,对于每个询问 l,r 输出[ ...
- 生成器模式(Builder)
1.本质:分离整体构建算法和部分构造 2.示意图: 3.功能: 构建复杂的产品,而且是细化的.分步骤的构建产品 分离构建算法和具体的构建实现 具体的构造实现可以方便的切换和扩展 4.优点: 1.松散耦 ...
- ios开发之IBOutlet和IBAction的区别
IBOutlet 输出口是使用关键字IBOutlet声明的实例变量.控制器头文件中的输出口声明应如下所示: @property (nonatomic, retain) IBOutlet UIButto ...
- Java实现Linux下服务器程序的双守护进程
作者:Vinkn 来自http://www.cnblogs.com/Vinkn/ 一.简介 现在的服务器端程序很多都是基于Java开发,针对于Java开发的Socket程序,这样的服务器端上线后出现问 ...
- JS作用域概念-预解析规则
// 作用域: // 域:空间.范围.区域…… // 作用:读.写 script 全局变量.全局函数 自上而下 函数 由里到外 {} 浏览器: “JS解析器” 1)“找一些东西” :var funct ...