【Foreign】猜测 [费用流]

猜测

Time Limit: 10 Sec  Memory Limit: 256 MB

Description

  

Input

  

Output

  

Sample Input

  3
  1 1
  1 2
  2 1

Sample Output

  3
  explain:
  (1,1),(1,1),(2,2)不是一个合法猜测(有相同的格子),因此不管怎么猜总是能全部猜中。

HINT

  

Main idea

  给定了若干个标准点,用这些点的横纵坐标分为x集和y集,定义猜点表示从x集和y集中各选一个,不能猜出重复的点,问在所有合法方案中最少包含上述几个标准点。

Source

  我们看到了这道题目,考虑从费用流的方法下手。

  我们从S->x集:容量为数字出现次数,费用为0y集->T:容量为数字出现次数,费用为0x集->y集:容量为1,若组合成了标准点则费用为1,否则为0

  然后我们这样连边,又由于题目要的是最少包含几个点,那么显然最小费用最大流就是答案了。

Code

  1 #include<iostream>
  2 #include<algorithm>
  3 #include<cstdio>
  4 #include<cstring>
  5 #include<cstdlib>
  6 #include<cmath>
  7 using namespace std;
  8
  9 const int ONE = 2000001;
 10 const int INF = 2147483640;
 11
 12 int n,x,y;
 13 int S,T;
 14 int E[1001][1001];
 15 int next[ONE],first[ONE],go[ONE],pas[ONE],Fro[ONE],tot=1;
 16 int from[ONE],q[1000001],dist[200001];
 17 bool vis[ONE];
 18 int tou,wei;
 19 int Ans,w[ONE];
 20 int li[ONE],li_num;
 21
 22 struct power
 23 {
 24         int x,y;
 25 }a[ONE],time[ONE],Max;
 26
 27 int get()
 28 {
 29         int res,Q=1;    char c;
 30         while( (c=getchar())<48 || c>57)
 31         if(c==‘-‘)Q=-1;
 32         if(Q) res=c-48;
 33         while((c=getchar())>=48 && c<=57)
 34         res=res*10+c-48;
 35         return res*Q;
 36 }
 37
 38 void Add(int u,int v,int liu,int z)
 39 {
 40         next[++tot]=first[u];   first[u]=tot;   go[tot]=v;  w[tot]=z;   pas[tot]=liu;   Fro[tot]=u;
 41         next[++tot]=first[v];   first[v]=tot;   go[tot]=u;  w[tot]=-z;  pas[tot]=0;     Fro[tot]=v;
 42 }
 43
 44 int Bfs()
 45 {
 46         memset(dist,63,sizeof(dist));
 47         dist[S]=0;  q[1]=S; vis[S]=1;
 48         tou=0;  wei=1;
 49         while(tou<wei)
 50         {
 51             int u=q[++tou];
 52             for(int e=first[u];e;e=next[e])
 53             {
 54                 int v=go[e];
 55                 if(dist[v]>dist[u]+w[e] && pas[e])
 56                 {
 57                     dist[v]=dist[u]+w[e]; from[v]=e;
 58                     if(!vis[v])
 59                     {
 60                         q[++wei]=v;
 61                         vis[v]=1;
 62                     }
 63                 }
 64             }
 65             vis[u]=0;
 66         }
 67         return dist[T]!=dist[T+10];
 68 }
 69
 70 void Deal()
 71 {
 72         int x=INF;
 73         for(int e=from[T];e;e=from[Fro[e]]) x=min(x,pas[e]);
 74         for(int e=from[T];e;e=from[Fro[e]])
 75         {
 76             pas[e]-=x;
 77             pas[e^1]+=x;
 78             Ans += w[e]*x;
 79         }
 80 }
 81
 82 int main()
 83 {
 84         n=get();
 85         for(int i=1;i<=n;i++)
 86         {
 87             a[i].x=get();    a[i].y=get();
 88             li[++li_num]=a[i].x; li[++li_num]=a[i].y;
 89         }
 90
 91         sort(li+1,li+li_num+1);
 92         li_num = unique(li+1,li+li_num+1) - li - 1;
 93         S=0;    T=2*li_num+1;
 94
 95         for(int i=1;i<=n;i++)
 96         {
 97             a[i].x = lower_bound(li+1,li+li_num+1, a[i].x) - li;
 98             a[i].y = lower_bound(li+1,li+li_num+1, a[i].y) - li;
 99             E[ a[i].x ][ a[i].y ] = 1;
100             time[a[i].x].x++;            time[a[i].y].y++;
101             Max.x = max(Max.x, a[i].x);    Max.y = max(Max.y, a[i].y);
102         }
103
104         for(int i=1;i<=Max.x;i++) if(time[i].x) Add(S,i,time[i].x,0);
105         for(int i=1;i<=Max.y;i++) if(time[i].y) Add(i+Max.x,T,time[i].y,0);
106
107         for(int i=1;i<=Max.x;i++)
108         if(time[i].x)
109         for(int j=1;j<=Max.y;j++)
110         if(time[j].y)
111         {
112             if(E[i][j]) Add(i,j+Max.x,1,1);
113             else Add(i,j+Max.x,1,0);
114         }
115
116         while(Bfs()) Deal();
117
118         printf("%d",Ans);
119
120 }

时间: 2024-08-06 07:58:22

【Foreign】猜测 [费用流]的相关文章

[BZOJ1070] [SCOI2007] 修车 (费用流 &amp; 动态加边)

Description 同一时刻有N位车主带着他们的爱车来到了汽车维修中心.维修中心共有M位技术人员,不同的技术人员对不同的车进行维修所用的时间是不同的.现在需要安排这M位技术人员所维修的车及顺序,使得顾客平均等待的时间最小. 说明:顾客的等待时间是指从他把车送至维修中心到维修完毕所用的时间. Input 第一行有两个m,n,表示技术人员数与顾客数. 接下来n行,每行m个整数.第i+1行第j个数表示第j位技术人员维修第i辆车需要用的时间T. Output 最小平均等待时间,答案精确到小数点后2位

【网络流24题】No.19 负载平衡问题 (费用流)

[题意] G 公司有 n 个沿铁路运输线环形排列的仓库, 每个仓库存储的货物数量不等. 如何用最少搬运量可以使 n 个仓库的库存数量相同.搬运货物时,只能在相邻的仓库之间搬运. 输入文件示例input.txt517 9 14 16 4 输出文件示例output.txt11 [分析] 其实我觉得这题可以贪心啊..n^2贪心??.没细想.. 打的是费用流.. 大概这样建图: 懒得写了..凌乱之美.. 求满流费用.. 1 #include<cstdio> 2 #include<cstdlib&

POJ 3422 kaka&#39;s matrix trvals(费用流)

#include <iostream> #include <cstring> #include <cstdio> #include <cstdlib> #include <algorithm> #include <vector> #include <queue> #include <stack> #include <set> #include <map> #include <cma

hdu 2448 Mining Station on the Sea【网络费用流】

Mining Station on the Sea Time Limit: 5000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 2371    Accepted Submission(s): 732 Problem Description The ocean is a treasure house of resources and the development

POJ训练计划3422_Kaka&#39;s Matrix Travels(网络流/费用流)

解题报告 题目传送门 题意: 从n×n的矩阵的左上角走到右下角,每次只能向右和向下走,走到一个格子上加上格子的数,可以走k次.问最大的和是多少. 思路: 建图:每个格子掰成两个点,分别叫"出点","入点", 入点到出点间连一个容量1,费用为格子数的边,以及一个容量∞,费用0的边. 同时,一个格子的"出点"向它右.下的格子的"入点"连边,容量∞,费用0. 源点向(0,0)的入点连一个容量K的边,(N-1,N-1)的出点向汇点连一

POJ 2135 Farm Tour &amp;&amp; HDU 2686 Matrix &amp;&amp; HDU 3376 Matrix Again 费用流求来回最短路

累了就要写题解,最近总是被虐到没脾气. 来回最短路问题貌似也可以用DP来搞,不过拿费用流还是很方便的. 可以转化成求满流为2 的最小花费.一般做法为拆点,对于 i 拆为2*i 和 2*i+1,然后连一条流量为1(花费根据题意来定) 的边来控制每个点只能通过一次. 额外添加source和sink来控制满流为2. 代码都雷同,以HDU3376为例. #include <algorithm> #include <iostream> #include <cstring> #in

【BZOJ3502/2288】PA2012 Tanie linie/【POJ Challenge】生日礼物 堆+链表(模拟费用流)

[BZOJ3502]PA2012 Tanie linie Description n个数字,求不相交的总和最大的最多k个连续子序列. 1<= k<= N<= 1000000. Sample Input 5 2 7 -3 4 -9 5 Sample Output 13 题解:跟1150和2151差不多. 我们先做一些预处理,因为连续的正数和连续的负数一定是要么都选要么都不选,所以可以将它们合并成一个数,同时区间中的零以及左右两端的负数没有意义,可以将它们删掉.然后我们得到的序列就变成:正-

POJ 2195 Going Home(费用流)

http://poj.org/problem?id=2195 题意: 在一个网格地图上,有n个小人和n栋房子.在每个时间单位内,每个小人可以往水平方向或垂直方向上移动一步,走到相邻的方格中.对每个小人,每走一步需要支付1美元,直到他走入到一栋房子里.每栋房子只能容纳一个小人. 计算出让n个小人移动到n个不同的房子需要支付的最小费用. 思路: 源点和每个人相连,容量为1,费用为0. 汇点和每栋房子相连,容量为1,费用为0. 每个人和每栋房子相连,容量为1,费用为人和房子之间的距离. 这样一来,跑一

洛谷P3381——费用流模板题

嗯..随便刷了一道费用流的模板题....来练练手. #include<iostream> #include<cstdio> #include<cstring> using namespace std; int h[5210],d[5210],used[5210],que[100010],last[5210]; int k=1,INF=0x7fffffff,ans1=0,ans2=0; inline int read(){ int t=1,num=0; char c=ge