标签:last 答案 pad 利用 one 它的 线段 namespace 数据
题意:
比方一条(x1,y) - (x2,y)的线段。我们在(x1,y)的位置上标记一个1。在(x2+1,y)的位置上标记一个-1,然后依照横坐标从左到右的顺序扫过这些点,事实上就是扫描线的思想了。然后在可行的范围内我们的P数组位置上+1。然后找出P中的最大值就可以。
这里在更新P数组的时候我们也不能够直接暴力更新,比方我们要更新区间[l,r]区间都+1。我们仅仅要在P[l]位置上+1,在P[r+1]的位置上-1,然后最后一起更新就可以。
代码例如以下:
#include <bits/stdc++.h> using namespace std; int N, M; pair<int, int> A[1000]; //储存点的坐标 struct event { //储存线段转化后的点,tp = 1 表示入点,tp = -1 表示出点 int x, y, tp; bool operator< (const event& other) const { return x < other.x; } }; event B[2015]; int P[2000016]; //维护答案的数组 int main() { //freopen("in.in", "r", stdin); //freopen("out.out", "w", stdout); scanf("%d%d", &N, &M); for (int i = 0; i < N; i++) { scanf("%d%d", &A[i].second, &A[i].first); //题目好坑啊,是依照y,x的顺序给的坐标 A[i].first += 1000000; } int y, x1, x2; for (int i = 0; i < M; i++) { scanf("%d%d%d", &y, &x1, &x2); B[i * 2] = (event) {x1, y, 1}; B[i * 2 + 1] = (event) {x2 + 1, y, -1}; } sort(B, B + 2 * M); //扫描前要先对线段转化的点依照横坐标进行排序 multiset<int> s; s.clear(); //s储存现在区间全部线段的纵坐标的集合 for (int i = 0; i < N; i++) { int last = -1; for (int j = 0, k; j < 2 * M; j = k) { for (k = j; k < 2 * M && B[k].x == B[j].x; k++) { if (B[k].tp == -1) s.erase(s.find(B[k].y)); else s.insert(B[k].y); } if (last != -1) P[A[i].first - B[j].x + 1]++, P[A[i].first - last + 1]--; //O(1)更新P数组 if (s.empty() || *s.begin() > A[i].second || *s.rbegin() < A[i].second) //表示现在横坐标下第i个点不能满足题意 last = -1; else last = B[j].x; //记录上次满足的横坐标 } } int ans = 0; for (int i = 1; i <= 2000015; i++) ans = max(ans, P[i] += P[i - 1]); //利用更新的信息同一时候寻找答案,即最大值 printf("%d\n", ans); return 0; }
Gym 100733J Summer Wars 题解:灵活运用扫描线的思想
标签:last 答案 pad 利用 one 它的 线段 namespace 数据
原文地址:http://www.cnblogs.com/slgkaifa/p/7045369.html