POJ 1161 Walls ( Floyd && 建图 )

题意 : 

在某国,城市之间建起了长城,每一条长城连接两座城市。每条长城互不相交。因此,从一个区域到另一个区域,需要经过一些城镇或者穿过一些长城。任意两个城市A和B之间最多只有一条长城,一端在A城市,另一端在B城市。从A走到B,可以只在一个区域内行走,或者只在长城上行走。

有一个俱乐部,它的会员分布在不同的城市中,每个城市要么只有一个会员,要么没有会员。会员们决定要集中到一个区域内聚会。他们骑车前往目的地。首先,由于城市内交通太堵,他们不想进入任何一个城市内,其次,他们希望穿越尽可能少的长城。他们确定一个聚会的区域之后,有的会员要穿越多条长城才能到达聚会区域,有的会员则可能本身就住在这个区域边上的城市所以不需穿越任何长城。他们希望找到一个区域,使得所有会员穿越长城的数量之和最小。

一共有N座城市,城市从1到N编号。如图一所示,有编号的顶点表示城市,每一条边表示一条长城。假设有3个会员,分别住在城市3、6和9,则最佳聚会区域和每个会员的路线如图二所示。共需穿越长城条数是2:城市9的会员需要穿越城市2和城市4之间的长城,城市6的会员需要穿越城市4和7之间的长城。给出城市、区域以及每个会员所在的城市,编一个程序,确定一个最佳的聚会区域,使得需要穿越长城的数量之和最小。

分析 :

这题先将各个区域抽象成点,然后每个区域之间如果有边相连 ( 例如题目所给图的 2、4、3、7 和 4、5、8 只通过了一个点相连,所以不算边相连 ) 那么就将这两个区域连接一条边权为 1 的点,然后对于各个城市如果它在某个区域的边缘上,那么就连一条城市到区域点且边权为 0 的有向边 ( 只是为了能让城市和区域之间有相连 ),最后根据这幅图去跑 Floyd 然后枚举区域点作为会员们聚集的点计算贡献取最小即可。这里的判断边相邻的方法,由于图的区域描述是按点顺时针给出(除了最后一个“外部区域”),所以可以根据这个特性,来用点的顺序描述一条边,比如开个数组 info[a][b] = 1 即 a 和 b 相连的边属于区域 1 ,如果下次再碰到有顺序 a、b 的边,那么说明此时这个区域和 1 边相连了,当然在判断的时候 info 里面的顺序是逆序的,而新判断的边是顺序的,至于为什么,模拟一下就知道了!

#include<stdio.h>
#include<algorithm>
#include<set>
#include<math.h>
using namespace std;
const int maxn = 500;
const int  INF = 0x3f3f3f3f;
int N, M, L;
int Members[30];
int tmp[maxn];
int info[maxn][maxn];///info[a][b] = c ==> 城市 a、b 之间的边属于区域 c
int dp[maxn][maxn];///跑 Floyd 的邻接矩阵
bool Belong[maxn][maxn];///Belong[a][b] = true ==> 区域 a 包含了点 b,false则反之,一开始初始化为 false

int main(void)
{
    while(~scanf("%d %d %d", &M, &N, &L)){
        for(int i=0; i<L; i++)
            scanf("%d", &Members[i]);

        for(int i=0; i<=M+N; i++){
            for(int j=0; j<=M+N; j++){
                if(i<=N && j<=N) info[i][j] = -1;
                if(i<=N && j<=M) Belong[i][j] = false;
                dp[i][j] = (i==j) ? 0 : INF;
            }
        }

        for(int i=1; i<=M; i++){
            int num;
            scanf("%d", &num);
            for(int j=0; j<num; j++){
                scanf("%d", &tmp[j]);
                Belong[i][tmp[j]] = true;
                //Belong[i].insert(tmp[j]);
                dp[tmp[j]+M][i] = 0; ///连一条城市到这个区域边权为 0 的边
            }

            //if(i == M) std::reverse(tmp, tmp+num);///最后的边是逆序的,一开始没想明白,其实如果倒序了就是错了
            for(int j=0; j<num; j++){
                int & Where = info[tmp[j]][tmp[(j+1)%num]];
                if(Where == -1) info[tmp[(j+1)%num]][tmp[j]] = i;///判断是否有边相连的情况,注意一下正顺和逆序
                else dp[i][Where] = dp[Where][i] = 1;
            }
        }

//        for(int i=1; i<=M+N; i++){
//            for(int j=1; j<=M+N; j++){
//                if(dp[i][j] == INF) printf("x ");
//                else printf("%d ", dp[i][j]);
//            }puts("");
//        }puts("");

        for(int k=1; k<=M+N; k++)
            for(int i=1; i<=M+N; i++)
                for(int j=1; j<=M+N; j++)
                    dp[i][j] = min(dp[i][j], dp[i][k]+dp[k][j]);

        int ans = INF;
        for(int i=1; i<=M; i++){///枚举集合的区域
            int sum = 0;
            for(int j=0; j<L; j++){
                if(Belong[i][Members[j]]) continue;
                sum += dp[Members[j]+M][i];///计算贡献
            }
            ans = min(ans, sum);
        }

        printf("%d\n", ans);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/Rubbishes/p/8309409.html

时间: 2024-08-07 10:53:20

POJ 1161 Walls ( Floyd && 建图 )的相关文章

POJ 1161 Walls(最短路+枚举)

POJ 1161 Walls(最短路+枚举) 题目背景 题目大意:题意是说有 n个小镇,他们两两之间可能存在一些墙(不是每两个都有),把整个二维平面分成多个区域,当然这些区域都是一些封闭的多边形(除了最外面的一个),现在,如果某几个小镇上的人想要聚会,为选择哪个区域为聚会地点,可以使他们所有人总共需要穿过的墙数最小,题目上有说明,不在某个点上聚会(聚会点在某个多边形内部),行进过程中不穿过图中的点(也就是除出发点外的其他小镇). 输入第1行代表m(2<=M<=200)个区域 第2行代表n(3&

POJ 1149 网络流 合并建图

这个题目我敲了一个简单的EK,这不是难点 难点在于建图,按题目的要求 每个猪圈和顾客都建点的话,那也太多了...我看了Edelweiss里面的缩点方法才建好的图,哎,惭愧啊 实际那些猪圈根本不需要单独建点,猪圈无非就是向顾客输送流量 以及向同时开着的猪圈输送流量,这一步可以直接缩为,当某个猪圈被第一次打开,它里面的流量就全部输送给那个顾客那个点,而且可以叠加,因为每一次猪圈是可以互通的,而且猪圈本身是没有容量限制,如果有限制,那就还得再考虑. 此外,每次对猪圈的接下来的访问者都进行建边.用来输送

POJ 2226 缩点建图+二分图最大匹配

这个最小覆盖但不同于 POJ 3041,只有横或者竖方向连通的点能用一块板子覆盖,非连续的,就要用多块 所以用类似并查集方法,分别横向与竖向缩点,有交集的地方就连通,再走一遍最大匹配即可 一开始还有点没想清楚缩点怎么写,其实就是横向和竖向分别缩一下,不要混在一起,否则很麻烦,要注意一下 #include <iostream> #include <cstdio> #include <cstring> using namespace std; char mat[900][9

POJ 1161 Walls【floyd 以面为点建图】

题目链接:http://poj.org/problem?id=1161 题目大意: 1.给出m个区域,n个俱乐部点.接下来是n个俱乐部点以及各个区域由什么点围成.求一个区域到各个俱乐部点的距离之和最小. 解题思路: 1.这题建图比较麻烦,以区域为点建图,区域之间若有边,则两区域的距离为1,建完图后跑一遍floyd就可以求出两两区域的最小距离. 2.对于答案,枚举每一个区域到各个俱乐部点的相邻区域的距离之和,取最小值. 1 #include<stdio.h> 2 #include<stri

poj 1161 Walls

https://vjudge.net/problem/POJ-1161 题意:有m个区域,n个小镇,有c个人在这些小镇中,他们要去某一个区域中聚会,从一个区域到另一个区域需要穿墙,问这些人聚到一起最少需要穿过几道墙.题中给出的区域是用小镇描述的,某几个小镇围成一个区域,每一个区域按照顺时针方向给出的.思路:首先用两个vector,一个描述一个区域,属性是这个区域的边界有哪些点,另一个描述一个点,属性是这些点属于哪个区域.然后接下来,我们把每一个区域看成一个点,开始建图.这题的难点就在于怎么建图,

poj 3281 最大流+建图

很巧妙的思想 转自:http://www.cnblogs.com/kuangbin/archive/2012/08/21/2649850.html 本题能够想到用最大流做,那真的是太绝了.建模的方法很妙! 题意就是有N头牛,F个食物,D个饮料. N头牛每头牛有一定的喜好,只喜欢几个食物和饮料. 每个食物和饮料只能给一头牛.一头牛只能得到一个食物和饮料. 而且一头牛必须同时获得一个食物和一个饮料才能满足.问至多有多少头牛可以获得满足. 最初相当的是二分匹配.但是明显不行,因为要分配两个东西,两个东

POJ 1161 Walls(Floyd , 建图)

题意: 给定n个城市, 然后城市之间会有长城相连, 长城之间会围成M个区域, 有L个vip(每个vip会处于一个城市里)要找一个区域聚会, 问一共最少跨越多少个长城. 分析: 其实这题难就难在建图, 因为图中的点不再是城市, 而是城市之间长城围成的区域, 只要把区域提取出来, 这题就是简单的Floyd了. 我们可以把每个区域的边先记录下来, 然后有共边的就是相邻的区域, vip所在的城市都是vip的出发区域,枚举聚会区域,然后都可能的vip区域都求一次取最少值再求和即可. #include<io

POJ 2112 最大流+二分+(建图:最重要的)

Optimal Milking Time Limit: 2000MS   Memory Limit: 30000K Total Submissions: 12521   Accepted: 4524 Case Time Limit: 1000MS Description FJ has moved his K (1 <= K <= 30) milking machines out into the cow pastures among the C (1 <= C <= 200) co

POJ 3686 The Windy&#39;s【最小权匹配(神建图啊)】

大意:有n个任务m个机器,告诉你n*m的矩阵表示每个任务在每个机器上完成需要的时间 问所有任务完成的总时间最少?(比如第一个任务在第一分钟完成第二个任务在第二分钟完成   则总时间为1 + 2 = 3 分析: 该题自己做的时候没有思路 后来在网上搜题解,感觉建图真是太厉害了 假设最优情况下,个个任务需要的时间分别为a1, a2, a3, ……,an 那么总时间t = n * a1 + (n - 1) * a2 + ……+ 2 * an - 1 + an 也就是说只需要考虑系数就可以了 我们先假设