码迷,mamicode.com
首页 > 其他好文 > 详细

「Codeforces 97B」Superset

时间:2020-05-12 10:08:37      阅读:62      评论:0      收藏:0      [点我收藏+]

标签:坐标   line   inline   构造   amp   const   codeforce   return   int()   

Description

定义一个点集是“好的”,那么其中任意两点之间满足下列三个条件之一:

  • 横坐标相同
  • 纵坐标相同
  • 以这两个点为顶点的长方形包含(在内部或边缘)其他的至少一个点。

现在有 \(n\) 个点,找到一种添加点的方案,使得这个集合是“好的”。

构造其中一组总点数不超过 \(2\times 10^5\) 的可行解。

Hint

  • \(1\le n\le 10^4\)
  • \(|\text{坐标大小}| \le 10^9\)

Solution

平面分治

假如将点按 \(x\) 坐标排序,取中间点的 \(x\) 坐标作为中间线的 \(x\) 坐标,然后对所有点做这个中间线上的 投影,所形成的点就是要添加的点。加入这些点之后,可以发现中间线异侧的点对全都满足条件。

那同侧的呢?使用 踢皮球大法 递归处理。于是做好了。如果要判断重复点的话只要用一个 set 就好了。

Code

#include <algorithm>
#include <iostream>
#include <set>

using namespace std;
const int N = 2e5 + 5;

struct Point {
	int x, y;
	inline Point() { }
	inline Point(int _x, int _y) : x(_x), y(_y) { }
	inline bool operator < (const Point& t) const {
		return x != t.x ? x < t.x : y < t.y;
	}
} p[N];
set<Point> points;
int n;

void solve(int l, int r) {
	if (l == r) return;
	int mid = (l + r) >> 1, x = p[mid].x;
	solve(l, mid), solve(mid + 1, r);
	
	for (register int i = l; i <= r; i++)
		points.insert(Point(x, p[i].y));
}

signed main() {
	ios::sync_with_stdio(false);
	
	cin >> n;
	for (register int i = 1; i <= n; i++)
		cin >> p[i].x >> p[i].y;
	for (register int i = 1; i <= n; i++)
		points.insert(p[i]);
	
	sort(p + 1, p + 1 + n);
	solve(1, n);
	
	cout << points.size() << endl;
	for (set<Point>::iterator it = points.begin(); it != points.end(); it++)
		cout << it->x << ‘ ‘ << it->y << endl;
	return 0;
}

「Codeforces 97B」Superset

标签:坐标   line   inline   构造   amp   const   codeforce   return   int()   

原文地址:https://www.cnblogs.com/-Wallace-/p/12874003.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!