题目描述
N(N<5000) 张矩形的海报,照片和其他同样形状的图片贴在墙上。它们的边都是垂直的或水平的。每个矩形可以部分或者全部覆盖其他矩形。所有的矩形组成的集合的轮廓称为周长。写一个程序计算周长。
图 1 是一个有 7 个矩形的例子:
图 1.一个 7 个矩形的集合对应的轮廓为图 2 所示的所有线段的集合:
图 2. 矩形集合的轮廓
所有矩形的顶点坐标均为整数。所有的坐标都在 [-10000,10000] 的范围内,并且任何一个矩形面积都为整数。结果的值可能需要 32 位有符号整数表示。
输入
第1行: N,张贴在墙上的矩形的数目
第 2..N+1行 接下来的N行中,每行都有两个点的坐标,分别是矩形的左下角坐标和右上角坐标。每一个坐标由 X 坐标和 Y 坐标组成。
输出
只有一行,为一个非负整数,表示输入数据中所有矩形集合的轮廓长度。
样例输入
7 -15 0 5 10 -5 8 20 25 15 -4 24 14 0 -6 16 4 2 15 10 22 30 10 36 20 34 0 40 16
样例输出
228
来源
IOI1998
其实这道题根本不需要什么覆盖层数,直接排序+暴力就能跑出0ms(洛谷0ms,FZOJ 4ms).
我们开一个数组highest,highest[i]存储[i, i+1)这条线段上所有已经搜过的矩形的最高的上边的高度。如果现在正在搜的矩形的最低边高于已经搜过的,则中间就会有一层没贴的,于是需要计入总周长。
怎么处理能使得搜到这个矩形时,所搜到的最高的上边的高度不再会覆盖掉这个矩形呢?其实只要根据矩形的下表面从下至上的顺序排序,这样就能保证以后搜到的矩形只会往上走,最高值只会越来越高,而不会突然又搜到下面的矩形,导致重复统计了。
时间复杂度O(n log n + n × (maxx - minx) + n × (maxy - miny))。如果加入离散,则最坏时间复杂度为O(n2)。
详见代码:
1 #include <cstdio> 2 #include <algorithm> 3 #include <cstring> 4 using namespace std; 5 6 //每个矩形的数据 7 struct data { 8 int lx,rx; //该矩形最左侧的横坐标与最右侧的横坐标 9 int ly,ry; //纵坐标同理 10 }; 11 12 const int maxn=5000, maxx=20000; 13 int n; 14 data a[maxn+1]; 15 int highest[maxx+1]; //已搜到的最高矩形上边界的高度 16 17 bool cmp_x(data x, data y) { return x.lx<y.lx; } 18 19 bool cmp_y(data x, data y) { return x.ly<y.ly; } 20 21 void kuaidu(int &p) { 22 char c; int f=1; p=0; 23 do { c=getchar(); if (c=='-') f=-1; } while (c<'0'||c>'9'); 24 do p=p*10+c-'0', c=getchar(); while (c>='0'&&c<='9'); 25 p*=f; 26 } 27 28 void init() { 29 kuaidu(n); 30 for (int i=1; i<=n; i++) { 31 kuaidu(a[i].lx); kuaidu(a[i].ly); kuaidu(a[i].rx); kuaidu(a[i].ry); 32 //使所有坐标变为正数 33 a[i].lx+=10001; a[i].ly+=10001; a[i].rx+=10001; a[i].ry+=10001; 34 } 35 } 36 37 int solve_x() { 38 memset(highest,-1,sizeof(highest)); 39 int ans=0; 40 //根据下底的高度,对所有矩形排序 41 sort(a+1,a+1+n,cmp_x); 42 for (int i=1; i<=n; i++) 43 for (int j=a[i].ly; j<a[i].ry; j++) { //处理区间[j,j+1) 44 //正在处理的矩形下边界高于已经搜到过的所有矩形的最高线 45 //所以在这个区间中多了一个矩形,答案加上上下边的长度(2) 46 if (highest[j]<a[i].lx) ans+=2; 47 //更新已搜到的矩形区间[j,j+1)的最高的线的高度 48 if (highest[j]<a[i].rx) highest[j]=a[i].rx; 49 } 50 return ans; 51 } 52 53 //y轴同理 54 int solve_y() { 55 memset(highest,-1,sizeof(highest)); 56 int ans=0; 57 sort(a+1,a+1+n,cmp_y); 58 for (int i=1; i<=n; i++) 59 for (int j=a[i].lx; j<a[i].rx; j++) { 60 if (highest[j]<a[i].ly) ans+=2; 61 if (highest[j]<a[i].ry) highest[j]=a[i].ry; 62 } 63 return ans; 64 } 65 66 int main() { 67 init(); 68 printf("%d ",solve_x()+solve_y()); 69 return 0; 70 }