티스토리 뷰

공부/Problem solving

성적 (USACO Jan 07 Gold)

구사과 2015. 11. 1. 13:36

https://www.acmicpc.net/problem/1859


O(N^2lgN)

Ti / Pi 증가 순으로 정렬된 시험 결과들을 토대로 avg(D)를 정의하자. avg(D)는 그리디 전략으로 D개의 과목을 드랍했을 때 베시의 점수이며, (T(N-D+1) + .... T(N)) / (P(N-D+1) + ... P(N)) 이다.

이 때 베시가 더 좋은 점수를 낼 조건은 Sum(T) / Sum(P) > avg(D) 를 만족하는 크기 D의 집합이 있다는 조건이다. 비슷한 유형을 풀어보면 알겠지만, 이는 Sum(T - avg(D) * P)  > 0 을 만족하는 크기 D의 집합이 있다는 것과 동치이다.

Sum(T - avg(D) * P) > 0을 만족하는 집합이 있는지... 는 굳이 말 안해도 다들 알겠지만, 1 ~ N 과목들을 쭉 돌면서 가장 저 값이 큰 D개의 과목을 그리디하게 가져가면 된다. 이는 NlgN의 정렬을 통해서 풀 수 있으니. D 하나당 쭉 시도해보면 N^2lgN에 답을 구할 수 있다.


O(NlgN)

avg(D) 값 자체는 O(N)에 모두 계산 가능한 값들이니 지금 시간 overhead는 Sum(T - avg(D) * P) > 0을 만족하는 D짜리 집합이 있는지를 판별하는 부분이라는 것을 알 수 있다. 저 함수를 빠르게 계산하는 것이 중요한 포인트다.

저런 걸 쉽게 해주는 data structure가 있을까? 일단 D = 1이라 쳐보자. T - x * P > 0을 만족하는 애가 있는지를 판별하는 건, Convex Hull Trick 스타일의 자료구조를 쓰면 알 수 있다는 걸 확인할 수 있다. 선분 집합에서 f(x)를 최대로 하는 게 무엇인지를 보면 된다. 하지만 D는 N까지도 올라갈 수 있기 때문에 일단 지금 저런 자료구조를 찾는건 의미가 없어 보인다.

여기서 짚고 넘어갈 것은 적어도 Sum(T - avg(D) * P) = 0을 만족하는 집합은 무조건 있다는 점이다. [N - D + 1, N] interval 안에 있는 과목이 바로 그를 만족하는 집합이 될 것이다. 그러면 Sum(T - avg(D) * P) != 0인지만 판별하면 되는데..! 이 말은 가장 큰 D개의 성적 중에서 [N - D + 1, N] interval 안에 없는 과목이 있다는 것을 뜻한다.

이제 D개의 선분의 합을 구할 필요 없이 깔끔하게 처리할 수 있는 방법을 낼 수 있다. [N - D + 1, N] interval 안에 있는 과목 중 T - x * P를 최소로 하는 과목이. [1, N - D] interval 안에 있는 과목 중 T - x * P를 최대로 하는 과목보다 값이 작으면, 해당 과목은 가장 큰 D개의 성적 안에 포함됨을 알 수 있다.

고로 이러한 고찰을 통해서 우리가 도달한 결론은

[N - D + 1, N] 구간의 Min(T - avg(D) * P) < [1, N - D] 구간의 Max(T - avg(D) * P)

를 만족하면 D를 찍으면 된다는 것이다. 이제 저걸 구하는 자료 구조가 필요한데 저건 convex hull trick으로 될법한 쿼리들이다. 두가지 방법을 소개하겠다.

일단 convex hull trick을 쓰기 전에 avg(D) <= avg(D+1)이라는 점을 짚고 넘어가겠다. 이 점을 짚고 넘어가지 못하면 첫번째 방법은 로그가 하나 더 붙고, 두번째 방법은 코드가 몇십줄이 늘어난다.

1. Segment Tree + CHT

이건 요즘 내가 자주 쓰는 방법인데 merge sort tree의 요령으로 각 노드가 구간의 선분을 convex hull trick으로 가지고 있도록 한다. (convex hull trick을 구조체로 미리 구현해 두는게 핵심) initialize는 O(NlgN)이다. 대충 짜면 O(Nlg^2N)이 될 수도 있음.

이제 들어오는 쿼리는 모두 구간 선분 최솟값 등을 물어보는 쿼리이니 lgN개의 노드에 대해서 대답해주면 된다. 쿼리의 단조성 덕에 이진 탐색이 필요하지 않다. 쿼리당 lgN이라 복잡도는 O(NlgN)

2. Sweeping + BST + CHT

위 방법보다 훨씬 직관적이고 빠르게 동작하지만, 코딩이 복잡한게 흠이다. 질의로 들어오는 구간이 너무 깔끔하기 때문에, 다음과 같은 방법을 쓴다.

1. CHT에 i번째 선분을 넣는다.

2. avg(i) 쿼리를 때린다.

CHT에 선분을 넣을 때 주의해야 할 점은 기울기에 단조성이 없다는 점이다. 그러한 상황에서 온라인 쿼리를 처리하려면 std::set을 통해서 동적으로 convex hull trick을 가지고 있어야 한다. 코딩 방법에 대해서는 생략하도록... 하겠다.

쿼리를 넣는 거는 일반 컨벡스 헐 트릭과 비슷하다. 데큐를 쓰듯이 앞을 잘라가면서 올라가면 amortized O(NlgN)에 풀 수 있다.


여담으로 "그냥" 컨벡스 헐로도 풀 수 있다고 한다. 그쪽은 생각을 안해봤는데 코드는 훨씬 짧다고 알고 있다. 관심이 있으면 http://contest.usaco.org/TESTDATA/JAN07.schul.htm 를 참고.

'공부 > Problem solving' 카테고리의 다른 글

Tourists (Codeforces 487E)  (2) 2015.11.15
Biconnected Component  (2) 2015.11.10
Cow Toll Paths (USACO Dec 09 Gold)  (0) 2015.11.01
Solving System of Difference Constraints  (3) 2015.10.18
순간이동 장치 (IOI 2008)  (0) 2015.10.15
댓글
공지사항
최근에 올라온 글
Total
Today
Yesterday