首页 > 其他分享 >扫描线

扫描线

时间:2024-06-11 22:45:38浏览次数:18  
标签:rt rs int rg ls 扫描线 define

扫描线

引入

扫描线一般运用在图形上面,它和它的字面意思十分相似,就是一条线在整个图上扫来扫去,它一般被用来解决图形面积、周长,以及二维数点等问题。

面积问题

例题1:【模板】扫描线

想象有一条线从下往上扫,会将整个图像依次扫描。我们只需要计算出每一条矩形(即图中同一颜色的小矩形)的面积,加起来就是整个图形的面积。

我们将每一条横边存下来并排序。将每个大矩形的下边标记为1,上边标记为-1,这样在扫描中只需要判断标记之和是否大于0即可知道当前扫描线上是否有图形覆盖。
每个大矩形的竖边延长交在x轴上(即图中的虚线),离散化后共有\(tot\)条直线,将有图形覆盖的地方分成\(tot-1\)个部分,我们使用线段树来维护每部分。
从下往上每扫描到一条横边,求出扫描线被图形覆盖的长度(即小矩形的长,\(len\)),乘以该条横边到下一条横边的距离(即小矩形的宽),就是这一条小矩形的面积了。
在\(pushup\)函数中,如果当前节点对应的部分被完全覆盖,那么就可以直接用右竖边坐标减去左竖边坐标算出该部分长度,直接返回,不必继续往下递归。而如果当前节点没有被完全覆盖,那么向下递归求出左右儿子对应部分的长度,加起来作为自己的长度。

#include<bits/stdc++.h>
#define rg register
#define qwq 0
#define ll long long 
#define ls rt << 1
#define rs rt << 1 | 1
using namespace std;
const int N = 1e6 + 3;
int n;
ll x, y, xx, yy, X[N << 1];
struct ScanLine {
	ll l, r, h;
	int flag;
	bool operator < (const ScanLine& ret) const { return h < ret.h; }
} line[N << 1];
struct SegmentTree {
	int l, r;
	int sum;  //是否被完全覆盖,不用bool是因为可能在-1前又有其它区间被完全覆盖
	ll len;  //扫描线在当前区间内被覆盖的长度
} t[N << 2];
inline void build(int rt, int l, int r) {
	t[rt].l = l;
	t[rt].r = r;
	if (l == r) return ;
	rg int mid = (l + r) >> 1;
	build(ls, l, mid);
	build(rs, mid + 1, r);
}
inline void pushup(int rt) {
	if (t[rt].sum) {  //被完全覆盖,可以直接相减得出长度 
		t[rt].len = X[t[rt].r + 1] - X[t[rt].l];
	} else {  //无法通过直接相减得出长度,便将左右子树的长度之和作为自己的长度
		t[rt].len = t[ls].len + t[rs].len; 
	}
}
inline void add(int rt, ll ql, ll qr, int val) {
	rg int l = t[rt].l, r = t[rt].r;  //该节点管辖的下标范围
	if (X[r + 1] <= ql || qr <= X[l]) return ;
	if (ql <= X[l] && X[r + 1] <= qr) {  //如果被完全覆盖,直接算 
		t[rt].sum += val;
		pushup(rt);
		return ;
	}
	add(ls, ql, qr, val);  //否则,先递归再加起来 
	add(rs, ql, qr, val);
	pushup(rt);
}
int main() {
	ios::sync_with_stdio(false);
	cin.tie(nullptr);
	cout.tie(nullptr);
	cin >> n;
	for (rg int i = 1; i <= n; i++) {
		cin >> x >> y >> xx >> yy;
		X[(i << 1) - 1] = x;
		X[i << 1] = xx;
		line[(i << 1) - 1] = {x, xx, y, 1};
		line[i << 1] = {x, xx, yy, -1};
	}
	n <<= 1;
	sort(line + 1, line + n + 1);
	sort(X + 1, X + n + 1);
	rg int tot = unique(X + 1, X + n + 1) - (X + 1);
	build(1, 1, tot - 1);
	rg ll ans = 0;
	for (rg int i = 1; i < n; i++) {  //最后一条边不用管,因为h为0
		add(1, line[i].l, line[i].r, line[i].flag);
		ans += t[1].len * (line[i + 1].h - line[i].h);  //统计面积
	}
	cout << ans << "\n";
	return qwq;
}

例题2:[poj1151]亚特兰蒂斯

同样是面积的并。

#include<bits/stdc++.h>
#define rg register
#define qwq 0
#define db double
using namespace std;
const int N = 1e4 + 3;
int n, cnt;
db x, y, xx, yy;
db X[N];
struct ScanLine {
	db l, r, h;
	int flag;
	bool operator < (const ScanLine& ret) { return h < ret.h; }
} line[N << 1];
#define ls rt << 1
#define rs rt << 1 | 1
struct SegmentTree {
	int l, r;
	int sum;
	db len;
} t[N << 2];
inline void build(int rt, int l, int r) {
	t[rt].l = l;
	t[rt].r = r;
	t[rt].sum = 0;  //这里记得要清空
	t[rt].len = 0;
	if (l == r) return ;
	rg int mid = (l + r) >> 1;
	build(ls, l, mid);
	build(rs, mid + 1, r);
}
inline void pushup(int rt) {
	if (t[rt].sum) {
		t[rt].len = X[t[rt].r + 1] - X[t[rt].l];
	} else {
		t[rt].len = t[ls].len + t[rs].len;
	}
}
inline void add(int rt, db ql, db qr, int val) {
	rg int l = t[rt].l, r = t[rt].r;
	if (X[r + 1] <= ql || qr <= X[l]) return ;
	if (ql <= X[l] && X[r + 1] <= qr) {
		t[rt].sum += val;
		pushup(rt);
		return ;
	}
	add(ls, ql, qr, val);
	add(rs, ql, qr, val);
	pushup(rt);
}
int main() {
	ios::sync_with_stdio(false);
	cin.tie(nullptr);
	cout.tie(nullptr);
	while (cin >> n && n) {
		for (rg int i = 1; i <= n; i++) {
			cin >> x >> y >> xx >> yy;
			X[(i << 1) - 1] = x;
			X[i << 1] = xx;
			line[(i << 1) - 1] = {x, xx, y, 1};
			line[i << 1] = {x, xx, yy, -1};
		}
		n <<= 1;
		sort(line + 1, line + n + 1);
		sort(X + 1, X + n + 1);
		rg int tot = unique(X + 1, X + n + 1) - (X + 1);
		build(1, 1, tot - 1);
		rg db ans = 0;
		for (rg int i = 1; i < n; i++) {
			add(1, line[i].l, line[i].r, line[i].flag);
			ans += t[1].len * (line[i + 1].h - line[i].h);
		}
		cout << "Test case #" << ++cnt << "\n";
		cout << "Total explored area: " << fixed << setprecision(2) << ans << "\n";
	}
	return qwq;
}

周长问题

例题3:[IOI1998] [USACO5.5] 矩形周长Picture

将图形周长分为两部分:横边和竖边。
先说竖边:

观察扫描线扫过的竖边,总结一下有这样一个结论:竖边总长度\(=\displaystyle\sum2\times\)被当前扫描线截得的整块数\(\times\)扫过的高度。(整块:第一条线上有1个块,第二条线上有3个块,第3条线上有2个块)
注意:如果出现两个高度相同的扫描线,也就是两矩形相邻,那么需要先扫底边再扫顶边,否则会多算。
对于横边:

可以得出结论:横边总长度\(=\displaystyle\sum |\)上次截得的总长\(-\)现在截得的总长\(|\)。
所以和面积比起来,周长并中的线段树还要维护线段的条数。另外,横边和竖边需要分别计算。

#include<bits/stdc++.h>
#define rg register
#define qwq 0
#define ls rt << 1
#define rs rt << 1 | 1
using namespace std;
const int N = 1e4 + 3;
int n;
int x, y, xx, yy, pre;
int X[N << 1];
struct ScanLine {
	int l, r, h, flag;
	bool operator < (const ScanLine& ret) {
		if (h == ret.h) return flag > ret.flag;  //先扫底边再扫顶边
		return h < ret.h;
	}
} line[N << 1];
struct SegmentTree {
	int l, r;
	int sum, len;
	int c;  //区间整块数 
	bool lf, rf;  //分别表示左、右端点是否被覆盖 
} t[N << 2];
inline void build(int rt, int l, int r) {
	t[rt].l = l;
	t[rt].r = r;
	if (l == r) return ;
	rg int mid = (l + r) >> 1;
	build(ls, l, mid);
	build(rs, mid + 1, r);
}
inline void pushup(int rt) {
	rg int l = t[rt].l, r = t[rt].r;
	if (t[rt].sum) {
		t[rt].len = X[r + 1] - X[l];
		t[rt].lf = t[rt].rf = true;
		t[rt].c = 1;
	} else {
		t[rt].len = t[ls].len + t[rs].len;
		t[rt].lf = t[ls].lf;  //如果左儿子左端点被覆盖,那么自己的左端点也肯定被覆盖 
		t[rt].rf = t[rs].rf;
		t[rt].c = t[ls].c + t[rs].c;
		if (t[ls].rf && t[rs].lf) t[rt].c -= 1;  //如果左儿子右端点和右儿子左端点都被覆盖,那么这是连续的一段,所以要-1
	}
}
inline void add(int rt, int ql, int qr, int flag) {
	rg int l = t[rt].l, r = t[rt].r;
	if (X[l] >= qr || X[r + 1] <= ql) return ;
	if (ql <= X[l] && X[r + 1] <= qr) {
		t[rt].sum += flag;
		pushup(rt);
		return ;
	}
	add(ls, ql, qr, flag);
	add(rs, ql, qr, flag);
	pushup(rt);
}
int main() {
	ios::sync_with_stdio(false);
	cin.tie(nullptr);
	cout.tie(nullptr);
	cin >> n;
	for (rg int i = 1; i <= n; i++) {
		cin >> x >> y >> xx >> yy;
		X[(i << 1) - 1] = x;
		X[i << 1] = xx;
		line[(i << 1) - 1] = {x, xx, y, 1};
		line[i << 1] = {x, xx, yy, -1};
	}
	n <<= 1;
	sort(line + 1, line + n + 1);
	sort(X + 1, X + n + 1);
	rg int tot = unique(X + 1, X + n + 1) - (X + 1);
	build(1, 1, tot - 1);
	rg int ans = 0;
	for (rg int i = 1; i < n; i++) {
		add(1, line[i].l, line[i].r, line[i].flag);
		ans += abs(pre - t[1].len);  //统计横边
		pre = t[1].len;
		ans += 2 * t[1].c * (line[i + 1].h - line[i].h);  //统计竖边 
	}
	ans += line[n].r - line[n].l;
	cout << ans << "\n";
	return qwq;
}

二维数点问题

例题4:窗口的星星

设某颗星星的坐标为\((x,y)\),则当窗户的右上角端点的坐标出现在\((x\sim x+w,y\sim y+h)\)这个范围内时,星星就会出现在窗户里。
因为题目中说出现在窗户边框的星星不算,我们不妨将框的长宽都减小0.5,所以坐标边界要-1,即\((x+w-1,y+h-1)\)。但我的写法本就没有竖边上的重合,因此为\((x+w,y+h-1)\)。
于是我们可以将每个星星都扩展成一个矩形,这时我们注意到,若两个矩形之间有交集,它们便可以放在同一个窗户中。

图中灰色部分就是两个星星构成的矩形的交集,只要窗户的右上角端点在灰色区域内,就能同时框住两个星星。
此时我们可以将问题转化为:平面上有若干个矩形,每个矩形都带有一个权值,求在哪个坐标上权值的总和最大。
我们直接将横边的权值设为星星的亮度,此时只需要求出扫描线上的区间最大值即可。
注意:对于上下两横边重合的情况,应先加入新边再删除旧边(同例题3)。

#include<bits/stdc++.h>
#define rg register
#define qwq 0
#define ll long long
#define ls rt << 1
#define rs rt << 1 | 1
using namespace std;
const int N = 2e4 + 3;
int T, n, W, H;
int x, y, l;
int X[N << 1];
struct ScanLine {
	int l, r, h;
	ll val;
	bool operator < (const ScanLine& ret) const {
		if (h == ret.h) return val > ret.val;
		return h < ret.h;
	}
} line[N << 1];
struct SegmentTree {
	int l, r;
	ll lazy;
	ll maxx;
} t[N << 2];
inline void build(int rt, int l, int r) {
	t[rt].l = l;
	t[rt].r = r;
	t[rt].lazy = t[rt].maxx = 0;
	if (l == r) return ;
	rg int mid = (l + r) >> 1;
	build(ls, l, mid);
	build(rs, mid + 1, r);
}
inline void pushup(int rt) {
	t[rt].maxx = max(t[ls].maxx, t[rs].maxx); 
}
inline void pushdown(int rt) {
	t[ls].maxx += t[rt].lazy;
	t[rs].maxx += t[rt].lazy;
	t[ls].lazy += t[rt].lazy;
	t[rs].lazy += t[rt].lazy;
	t[rt].lazy = 0;
}
inline void add(int rt, int ql, int qr, int val) {
	rg int l = t[rt].l, r = t[rt].r;
	if (X[r + 1] <= ql || qr <= X[l]) return ;
	if (ql <= X[l] && X[r + 1] <= qr) {
		t[rt].lazy += val;
		t[rt].maxx += val;
		return ;
	}
	pushdown(rt);
	add(ls, ql, qr, val);
	add(rs, ql, qr, val);
	pushup(rt);
}
int main() {
	ios::sync_with_stdio(false);
	cin.tie(nullptr);
	cout.tie(nullptr);
	cin >> T;
	while (T--) {
		cin >> n >> W >> H;
		for (rg int i = 1; i <= n; i++) {
			cin >> x >> y >> l;
			X[(i << 1) - 1] = x;
			X[i << 1] = x + W;
			line[(i << 1) - 1] = {x, x + W, y, l};
			line[i << 1] = {x, x + W, y + H - 1, -l};
		}
		n <<= 1;
		sort(line + 1, line + n + 1);
		sort(X + 1, X + n + 1);
		rg int tot = unique(X + 1, X + n + 1) - (X + 1);
		build(1, 1, tot - 1);
		rg int ans = 0;
		for (rg int i = 1; i < n; i++) {
			add(1, line[i].l, line[i].r, line[i].val);
			ans = max(ans, t[1].maxx);
		}
		cout << ans << "\n";
	}
	return qwq;
}

标签:rt,rs,int,rg,ls,扫描线,define
From: https://www.cnblogs.com/Baiyj/p/18242909

相关文章

  • 扫描线模板
    #include<bits/stdc++.h>usingnamespacestd;usingi64=longlong;constintN=1e6+5;//本模板是从左往右扫的,从下往上扫同理#definels(rt<<1)#definers(rt<<1|1)i64cover[N*8];//存放i节点对应区间覆盖情况的值i64n;i64len[N*8];i64yy[N*2];/......
  • 扫描线
    题目链接https://leetcode.cn/problems/rectangle-area-ii/题目大意题目思路选取连续的x值:(left,right),在这个区间内,沿着x轴的方向扫描,求出所有符合条件的(y1,y2)算出扫描区间的h,结合w*h,算出面积!礼貌拿图,多谢三叶姐(https://leetcode.cn/problems/rectangle-area-ii/so......
  • 扫描线的应用1
    概述顾名思义,扫描线通常是在二维空间内,模拟一条线上下扫描,以此达到解决求二维空间的矩形面积或点数问题。实现方法通常采用排序,离散化等操作。排序是为了扫描的不重不漏,而离散化是由于我们模拟的扫描线会一段段扫描,可以去掉重复的信息。 矩形面积1151--Atlantis给定平......
  • 扫描线
    扫描线是什么&矩形面积并本质上可以理解为,对一个二维(或多维)平面上的问题,通过扫描一维,并且用数据结构动态维护另一位的增减性。我们对于\(x\)轴扫描,那么会对面积构成影响的\(y\)显然只有红色的几条和绿色的几条。容易发现,这样的线的数量是\(2n\)的(\(n\)为矩形个数)。......
  • 换维扫描线
    简介一般来说,我们处理某些可以离线的问题,我们会将询问离线,然后将修改挂在左端点或右端点,然后从左往右扫描这些修改,并处理询问,数据结构记录的一般是下标\(i\)到当前走到的地方的一些信息。而换维扫描线则采取了截然相反的措施:我们将区间修改转化成差分,然后从左往右扫描序列,线段......
  • 计算几何——扫描线 学习笔记
    计算几何——扫描线学习笔记你会发现我的笔记的顺序和很多扫描线的讲解是反着来的。其实是和我老师给的课件完全是逆序(谁帮我算一下逆序对啊喵)。前言一开始以为扫描线就是用来求二维几何图像的信息的。但是其实这个并不准确。个人认为,扫描线其实是一个思想,就像动态规划一样......
  • 扫描线学习笔记
    1.引入扫描线多用于图形上,是一条线在图形上扫来扫去,它一般被用来解决图形面积,周长,以及二维数点等问题。2.扫描线求面积并如下图:我们模拟一条扫描线,使它从下往上扫过整个平面,这条扫描线会在遇到横向线段的时候停下来更新一些东西。那么整个图形就可以找出四条线段,如图:更新的......
  • 【学习笔记】扫描线
    bilibili:BV1Mm411D7kr讲了一下。模板代码:面积并:#include<cstring>#include<iostream>#include<algorithm>#defineintlonglongusingnamespacestd;namespaceIO{template<typenameT>Tread(Tx){Tsum=0,opt=1......
  • 扫描线
    扫描线的精髓是通过离线、引入时间维,把静态询问降低一维、变成动态问题。一般是先把若干询问通过可减性变成前缀询问,再离线、从左到右排序,从左到右一个一个地一边加入元素,一边回答询问。以下是例题+一句话题解。(虽然可能不是真的一句话)平面最值二维平面上\(n(\le10^5)\)个......
  • 线段树维护扫描线
    你需要实现一种数据结构支持以下操作:区间加减1保证加减区间一一对应,且先加后减,序列中永远不出现负数。查询完整序列中0的个数#include<cstdio>#include<algorithm>constintmaxn=1e5+10;longlongx[maxn*2];structsegmentTree{ structnode { i......