[ SDOI 2006 ] 仓库管理员的烦恼
\(\\\)
Description
有 \(n\) 种货物和 \(n\) 个仓库,开始第 \(i\) 个仓库里有 \(a_{ij}\) 个第 \(j\) 种货物。
现在要让每种货物都只放到一个仓库里,且一个仓库只放一种货物。
货物将在仓库之间移动,总代价就是移动的所有货物的总重。
现不指定哪种货物放到哪个仓库里,求最小总代价。
- \(n\le 150,a_{ij}\le 100\)
\(\\\)
Solution
一开始的想法是把限制加到流量上,后来发现不行。
为什么呢?首先这题的建图肯定是一侧货物一侧仓库。
如果是货物 \(\to\) 仓库,我们无法确定最后哪个仓库放哪种货物,所以仓库向汇点的流量无法限制。
如果是仓库 \(\to\) 货物,我们又无法确定每个仓库向每个货物的流量了,因为我们不知道这种货物是否放回到这个仓库里,进而代价会算多 \((\) 有的边费用可能是 \(0\) 没有算上 \()\) 。
\(\\\)
因此我们要把代价加到费用上。
考虑此题的限制:
- 每种货物只能放到一个仓库里
- 每个仓库只能放一种货物
于是原点连出和连入汇点的所有边流量都是 \(1\) ,代表如上限制。
我们预处理 \(sum_i\) 表示第 \(i\) 种货物的总量。
\(\\\)
如果源连货物,货物指向仓库,仓库连汇,则建图方式:
因为是货物向仓库连边,那么如果这一货物要放到这一仓库,原本在这一仓库的此货物就不用产生代价,其他的这一货物都要产生代价。
货物 \(j\) 向仓库 \(i\) 要连一条流量为 \(1\) ,费用为 \(sum_j-a_{ij}\) 的边。
\(\\\)
如果源连仓库,仓库指向货物,货物连汇,则建图方式:
此时所谓的货物,其实是代表的存放这一货物的仓库。
而仓库连向货物,代价就应该是令这一仓库作为存放这一货物的代价。
显然费用和原来思考方式相同。
\(\\\)
Code
将第二种建图方式的代码放在了注释里。
#include<cmath>
#include<queue>
#include<cstdio>
#include<cctype>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 310
#define M 50010
#define R register
#define gc getchar
#define inf 1000000000
using namespace std;
inline int rd(){
int x=0; bool f=0; char c=gc();
while(!isdigit(c)){if(c=='-')f=1;c=gc();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();}
return f?-x:x;
}
int n,m,s,t,tot=1,maxn;
int hd[N],pre[N],id[N],dis[N],w[N][N],sum[N];
struct edge{int f,w,to,nxt;}e[M];
inline void add(int u,int v,int f,int w){
e[++tot].to=v; e[tot].w=w;
e[tot].f=f; e[tot].nxt=hd[u]; hd[u]=tot;
}
bool vis[N];
queue<int> q;
inline bool spfa(){
for(R int i=1;i<=maxn;++i) dis[i]=inf,vis[i]=0;
dis[s]=0; q.push(s);
while(!q.empty()){
int u=q.front();
q.pop(); vis[u]=0;
for(R int i=hd[u],v;i;i=e[i].nxt)
if(e[i].f&&(dis[v=e[i].to]>dis[u]+e[i].w)){
dis[v]=dis[u]+e[i].w;
pre[v]=u; id[v]=i;
if(!vis[v]) vis[v]=1,q.push(v);
}
}
return dis[t]<inf;
}
inline int mcmf(){
int res=0,tmp;
while(spfa()){
tmp=inf;
for(R int i=t;i!=s;i=pre[i]) tmp=min(tmp,e[id[i]].f);
for(R int i=t;i!=s;i=pre[i]){
e[id[i]].f-=tmp; e[id[i]^1].f+=tmp;
}
res+=tmp*dis[t];
}
return res;
}
int main(){
n=rd();
s=0; maxn=t=(n<<1)+1;
for(R int i=1;i<=n;++i){
add(s,i,1,0); add(i,s,0,0);
add(n+i,t,1,0); add(t,n+i,0,0);
for(R int j=1;j<=n;++j) w[i][j]=rd();
}
for(R int i=1;i<=n;++i)
for(R int j=1;j<=n;++j) sum[j]+=w[i][j];
for(R int i=1;i<=n;++i)
for(R int j=1;j<=n;++j){
add(j,n+i,1,sum[j]-w[i][j]);
add(n+i,j,0,w[i][j]-sum[j]);
}
/*
for(R int i=1;i<=n;++i)
for(R int j=1;j<=n;++j){
add(i,n+j,1,sum[j]-w[i][j]);
add(n+j,i,0,w[i][j]-sum[j]);
}
*/
printf("%d\n",mcmf());
return 0;
}
[ SDOI 2006 ] 仓库管理员的烦恼的更多相关文章
- [SDOI2006]仓库管理员的烦恼
题目描述 仓库管理员M最近一直很烦恼,因为他的上司给了他一个艰难的任务:让他尽快想出一种合理的方案,把公司的仓库整理好. 已知公司共有n个仓库和n种货物,由于公司进货时没能很好的归好类,使得大部分的仓 ...
- 解题:ZJOI 2006 皇帝的烦恼
禁止DP,贪心真香 有一个比较明显的贪心思路是让每个人和距离为$2$(隔着一个人)的人尽量用一样的,这样只需要扫一遍然后对每对相邻的人之和取最大值即可.但是当人数为奇数时这样就会出锅,因为最后一个人和 ...
- [ SDOI 2006 ] 保安站岗
\(\\\) Description 给出一棵 \(n\) 个节点以 \(1\) 为根的树,一个节点的覆盖半径是 \(1\) ,点有点权 \(val_x\) . 选择一些点,使得点权和最小,同时每个节 ...
- [SDOI2006] 仓库管理员的烦恼 - 二分图最大权匹配
最小化代价,即最大化"本土"货物的数量 于是就是个二分图最大权匹配裸题 #include <bits/stdc++.h> using namespace std; #d ...
- 【热门技术】EventBus 3.0,让事件订阅更简单,从此告别组件消息传递烦恼~
一.写在前面 还在为时间接收而烦恼吗?还在为各种组件间的消息传递烦恼吗?EventBus 3.0,专注于android的发布.订阅事件总线,让各组件间的消息传递更简单!完美替代Intent,Handl ...
- BZOJ 2006: [NOI2010]超级钢琴
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MBSubmit: 2613 Solved: 1297[Submit][Statu ...
- CPU阿甘之烦恼
转自“码农翻身”公共号,原文地址CPU阿甘之烦恼 总结:(程序加载到内存运行的演变过程) 内存存放程序.OS负责加载程序到内存.CPU负责运行内存中的程序 1.串行:加载一个完整程序到内存,CPU运行 ...
- BZOJ 1005 [HNOI2008] 明明的烦恼(组合数学 Purfer Sequence)
题目大意 自从明明学了树的结构,就对奇怪的树产生了兴趣...... 给出标号为 1 到 N 的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为 N( ...
- [bzoj1935][shoi2007]Tree 园丁的烦恼(树状数组+离线)
1935: [Shoi2007]Tree 园丁的烦恼 Time Limit: 15 Sec Memory Limit: 357 MBSubmit: 980 Solved: 450[Submit][ ...
随机推荐
- 权限框架之Shiro详解(非原创)
文章大纲 一.权限框架介绍二.Shiro基础介绍三.Spring Boot整合Shiro代码实战四.项目源码与资料下载五.参考文章 一.权限框架介绍 1. 什么是权限管理 权限管理属于系统安全 ...
- Systemtap工具查看系统资源使用
http://www.xbwolf.com/507 http://blog.csdn.net/kafeiflynn/article/details/6429976 安装步骤yum install sy ...
- OCP知识点讲解 之 队列、资源与锁:RHCA|OCM|CCIE RedHat大中华地区前50位RHCA系统架构师:叶绍琛
一.队列与共享资源 共享资源可以被多个会话.进程同时访问,因此它的访问需要保护.Oracle中,除了PGA,所有的东西(包括内存.磁盘.CPU.表.索引.事务等等,种类太多,一概用东西两字来代表) ...
- JavaSE入门学习5:Java基础语法之keyword,标识符,凝视,常量和变量
一keyword keyword概述:Java语言中有一些具有特殊用途的词被称为keyword.keyword对Java的编译器有着特殊的意义.在程 序中应用时一定要谨慎. keyword特点:组成k ...
- 任务调度(三)——Timer的替代品ScheduledExecutorService简单介绍
先前的两篇博文<任务调度(一)--jdk自带的Timer>和<任务调度(二)--jdk自带的Timer 动态改动任务运行计划>中,简介了一下Timer,能够实现几本的功能.可是 ...
- .NET Core/.NET之Stream简介 Rx.NET 简介
.NET Core/.NET之Stream简介 之前写了一篇C#装饰模式的文章提到了.NET Core的Stream, 所以这里尽量把Stream介绍全点. (都是书上的内容) .NET Core ...
- 模式识别之ocr---文字识别Tesseract-OCR 进行文字识别 VS2010
近日做铸件文字识别的项目,需要识别铸件上的字符和数字,找到开源的识别库Tesseract,下面简单记录下怎么使用. 首先在项目主页http://code.google.com/p/tesseract- ...
- android 3G移植【转】
本文转载自:http://blog.csdn.net/hanmengaidudu/article/details/17028383 一 开发环境简介 内容 说明 3G模块 华为EM820W(WCDMA ...
- POJ2115 C-Loop
传送门 这道题是求解不定方程的一道好练习题. 题目描述的很诡异……还说什么k进制,其实就是要求一个数A,每次加C,问到B要加多少次,所有的数对2k取模. 也就是说我们能列出如下方程:A+xC ≡ B ...
- 用C#读取txt文件的方法(转)
.使用FileStream读写文件 文件头: using System; using System.Collections.Generic; using System.Text; using Syst ...