코딩/백준 문제 풀이

BOJ 12876 - 반평면 땅따먹기 2

stonejjun 2021. 4. 12. 15:48

문제 링크 : www.acmicpc.net/problem/12876

문제 태그

문제 풀이

보통 반평면 땅따먹기 1은 풀고 올 것이니 이에 대한 풀이는 알고 있다고 가정하려고 한다. 반평면 땅따먹기 1은 그냥 일반적인 cht optimization이 아니라 직선의 기울기의 단조성이 없기 때문에 lichao tree를 써야하는 문제였다. 반평면 땅따먹기 2에서는 이제 선분이 중간에 없어진다. 

선분에 life가 존재하게 되었다 이러한 문제는 offline dynamic connectivity로 풀 수 있다. lichao tree에서 선분이 추가될 때, 노드를 내려가면서 여러개의 노드를 업데이트 하게 된다. 그 과정에서 노드의 번호와, 업데이트 하기전 노드의 상태를 저장하고 있다가, 그 번호의 노드를 저장한 상태로 되돌리는 것으로 undo 과정을 진행하면 된다. 이때 한 번의 선분추가로 인해 변화된 노드들을 묶어서 한번에 저장하고, 한 번의 undo 과정에서 다 같이 변화시켜야 할 것이다. 

코드

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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
#include<bits/stdc++.h>
using namespace std;
typedef long long int ll;
typedef pair<ll,ll> pii;
 
#define ff first
#define ss second
#define pb push_back
#define eb emplace_back
#define mp make_pair
ll inf=2e18+7;
 
struct lin{
    ll a,b;
    ll sol(ll x){
        return a*x+b;
    }
};
 
typedef pair<lin,ll> pli;
typedef pair<pli,ll> plii;
 
vector<plii> stac;
ll ccnt;
 
struct node{
    int l,r;
    ll s,e;
    lin line;
};
 
vector<node> tree;
void init(ll s,ll e){
    tree.push_back({-1,-1,s,e,{0,-inf}});    
}
 
void upt(ll nod,lin nl){
    stac.pb({{tree[nod].line,nod},ccnt});
    ll s=tree[nod].s;
    ll e=tree[nod].e;
    ll m=(s+e)/2;
    lin l1=tree[nod].line;
    lin l2=nl;
    if(l1.sol(s)>l2.sol(s)) swap(l1,l2);
    if(l1.sol(e)<=l2.sol(e)){
        tree[nod].line=l2;
        return ;
    }
    if(l1.sol(m)>=l2.sol(m)){
        tree[nod].line=l1;
        if(tree[nod].l==-1){
            tree[nod].l=tree.size();
            init(s,m);
        }
        upt(tree[nod].l,l2);
    }
    else{
        tree[nod].line=l2;
        if(tree[nod].r==-1){
            tree[nod].r=tree.size();
            init(m+1,e);
        }
        upt(tree[nod].r,l1);
    }
}
 
ll qry(ll nod,ll x){
    if(nod==-1return -inf;
    if(x<=(tree[nod].s+tree[nod].e)/2
        return max(tree[nod].line.sol(x),qry(tree[nod].l,x));
    else 
        return max(tree[nod].line.sol(x),qry(tree[nod].r,x));
}
 
void undo(){
    ll sz=stac.size();
    ccnt--;
    while(sz&&stac[sz-1].ss>ccnt){
        auto k=stac[sz-1];
        tree[k.ff.ss].line=k.ff.ff;
        stac.pop_back();
        sz--;
    }
}
 
ll ans[302020];
 
struct qnode{
    vector<lin> eds;
    ll sol,p1; 
};
qnode qtree[1210101];
 
void uptl(lin ed,ll l,ll r,ll s,ll e,ll nod){
    if(r<s||e<l) return;
    if(l<=s&&e<=r){
        qtree[nod].eds.pb(ed);
        return;
    }
    uptl(ed,l,r,s,(s+e)/2,nod*2);
    uptl(ed,l,r,(s+e)/2+1,e,nod*2+1);
}
 
void chk(ll idx,ll pp1,ll s,ll e,ll nod){
    if(idx<s||idx>e) return ;
    if(s==e){
        qtree[nod].sol=1;
        qtree[nod].p1=pp1;
        return ;
    }
    chk(idx,pp1,s,(s+e)/2,nod*2);
    chk(idx,pp1,(s+e)/2+1,e,nod*2+1);
}
 
void dfs(ll s,ll e,ll nod){
    ll sz=qtree[nod].eds.size();
    for(ll i=sz-1;i>=0;i--){
        ccnt++;
        auto k=qtree[nod].eds[i];
        upt(0,k);
    }
    if(s==e){
        if(qtree[nod].sol)
            ans[s]=qry(0,qtree[nod].p1);
        for(auto k:qtree[nod].eds)
            undo();
        return ;
    }
 
    dfs(s,(s+e)/2,nod*2);
    dfs((s+e)/2+1,e,nod*2+1);
    for(auto k:qtree[nod].eds)
        undo();
    
    return ;
}
vector<pli> v;
ll ch[303031];
int main(){
    ll i,j,k,l,m,n,a,b,c;
    scanf("%lld",&n);
    init(-1e9-7,1e9+7);
    v.resize(n+2);
    for(i=1;i<=n;i++){
        v[i].ss=n;
        scanf("%lld",&a);
        if(a==1){
            scanf("%lld %lld",&b,&c);
            v[i].ff.a=b;
            v[i].ff.b=c;
            ch[i]=1;
        }
        else if(a==2){
            scanf("%lld",&b);
            v[b].ss=i;
        }
        else{
            scanf("%lld",&b);
            ch[i]=3;
            chk(i,b,1,n,1);
        }
    }
    for(i=1;i<=n;i++)
        if(ch[i]==1)
            uptl(v[i].ff,i,v[i].ss,1,n,1);
    
    dfs(1,n,1);
    for(i=1;i<=n;i++)
        if(ch[i]==3){
            if(ans[i]<=-1e18-5e17printf("EMPTY\n");
            else printf("%lld\n",ans[i]);
        }
}
 
cs

'코딩 > 백준 문제 풀이' 카테고리의 다른 글

BOJ 22847 공식풀이의 확장  (0) 2021.10.02
BOJ 18721 - clique  (0) 2021.05.15
IOI 2018 day1 - Combo  (0) 2021.03.12
BOJ 11738 - Distance on Triangulation  (0) 2021.03.10
BOJ 3408 - Non-boring sequences  (0) 2021.02.15