티스토리 뷰
문제
https://oj.uz/problem/view/JOI19_examination
문제 보기 - Examination (JOI19_examination) :: oj.uz
문제 보기 - Examination (JOI19_examination)
oj.uz
N개의 $(x, y)$점이 들어오고, Q개의 $(a, b, c)$쿼리가 들어온다. 쿼리의 의미는 $x>=a, y>=b, x+y>=c$를 만족하는 점들의 수를 세는 것이다.
$N<=100000$
$Q<=100000$
$x, y, a, b, c<=1000000000$
풀이
$c=0$이라면 단순히 직사각형 속의 점들의 수를 세는 문제이니, 2D Segment Tree를 이용하거나, 분할정복을 이용하여 $O(Nlog^2N)$ 안에 해결할 수 있다.
$c$에 대한 조건을 해결하기 위한 방법은 시간축을 도입하는 것으로, $x+y$와 $c$로 전체 점과 쿼리를 정렬하여 각 쿼리마다 $x+y>=c$를 만족하는 점들만 업데이트한 후 똑같은 직사각형 처리를 하는 것이다.
결론적으로 점 추가 업데이트가 있고, 직사각형 속의 점들의 수를 세는 문제이니, 가장 효율적인 방법은 분할정복을 이용하여 펜윅 트리로 스위핑을 하여 $O(Nlog^2N)$에 해결하는 것이다.
시간 복잡도 : $O((N+Q)log^2(N+Q))$
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int, int> pii;
typedef pair<ll, ll> pll;
const int MAXN = 1e5;
struct Data
{
int x, y, p, s;
};
int N, Q;
vector<Data> V;
vector<int> comp;
int ans[MAXN+10];
int getcomp(int x) { return lower_bound(comp.begin(), comp.end(), x)-comp.begin()+1; }
int tree[MAXN+10];
void update(int i, int k) { for(i=N-i+1; i<=N; i+=(i&-i)) tree[i]+=k; }
int query(int i) { int ret=0; for(i=N-i+1; i>0; i-=(i&-i)) ret+=tree[i]; return ret; }
void flush(int i) { for(i=N-i+1; i<=N; i+=(i&-i)) tree[i]=0; }
void solve(int l, int r)
{
if(l==r) return;
int mid=l+r>>1;
solve(l, mid);
solve(mid+1, r);
vector<Data> L, R;
for(int i=l; i<=mid; i++) if(V[i].p==0) L.push_back(V[i]);
for(int i=mid+1; i<=r; i++) if(V[i].p) R.push_back(V[i]);
sort(L.begin(), L.end(), [&](const Data &p, const Data &q) { return p.x>q.x; });
sort(R.begin(), R.end(), [&](const Data &p, const Data &q) { return p.x>q.x; });
for(int i=0, j=0; i<R.size(); i++)
{
for(; j<L.size() && L[j].x>=R[i].x; j++)
{
update(getcomp(L[j].y), 1);
}
ans[R[i].p]+=query(getcomp(R[i].y));
}
for(int j=0; j<L.size(); j++) flush(getcomp(L[j].y));
}
int main()
{
scanf("%d%d", &N, &Q);
for(int i=1; i<=N; i++)
{
int x, y;
scanf("%d%d", &x, &y);
V.push_back({x, y, 0, x+y});
comp.push_back(y);
}
for(int i=1; i<=Q; i++)
{
int x, y, z;
scanf("%d%d%d", &x, &y, &z);
V.push_back({x, y, i, z});
}
sort(comp.begin(), comp.end());
comp.erase(unique(comp.begin(), comp.end()), comp.end());
sort(V.begin(), V.end(), [&](const Data &p, const Data &q)
{
if(p.s!=q.s) return p.s>q.s;
return p.p<q.p;
});
solve(0, V.size()-1);
for(int i=1; i<=Q; i++) printf("%d\n", ans[i]);
}
'JOISC > 2019' 카테고리의 다른 글
JOISC19 Dishes (0) | 2020.09.10 |
---|---|
JOISC19 Antennas (0) | 2020.09.08 |
JOISC19 Naan (0) | 2020.09.07 |
댓글
공지사항
최근에 올라온 글
최근에 달린 댓글
- Total
- Today
- Yesterday
링크
TAG
- CHT
- graph
- ioi
- Interactive
- convex hull
- tree
- Lazy Propagation
- Floyd-Warshall
- Parametric Search
- BOJ
- Line sweeping
- Sparse Table
- DP
- HLD
- Fenwick Tree
- Greedy
- Divide & Conquer
- Union Find
- stack
- Codeforces
- ⭐
- APIO
- Merge Sort
- Centroid Decomposition
- Segment Tree
- offline
- Persistent Segment Tree
- DFS
- Shortest path
- Sqrt Decomposition
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | 5 | ||
6 | 7 | 8 | 9 | 10 | 11 | 12 |
13 | 14 | 15 | 16 | 17 | 18 | 19 |
20 | 21 | 22 | 23 | 24 | 25 | 26 |
27 | 28 | 29 | 30 |
글 보관함