이 제출은 이전 버전의 oj.uz에서 채점하였습니다. 현재는 제출 당시와는 다른 서버에서 채점을 하기 때문에, 다시 제출하면 결과가 달라질 수도 있습니다.
// O(NlogN)
#include <algorithm>
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
typedef long long ll;
typedef vector<int> vi;
class Segment_Tree{
private:
int n;
vi a,b;
void Max(int &x,int y){
if(x<=y||!b[x]) x=y;
}
public:
Segment_Tree(int n_){
n=1;
while(n<n_) n*=2;
a=vi(2*n-1,-1);
b=vi(n_,1);
}
void ChangeMax(int l,int r,int id){
l+=n-1;r+=n-1;
while(l<r){
if(l%2==0){
Max(a[l],id);
l++;
}
if(r%2==0){
r--;
Max(a[r],id);
}
l/=2;r/=2;
}
}
void Erase(int x){b[x]=0;}
int Access(int k){
k+=n-1;
int m=a[k];
while(k>0){
k=(k-1)/2;
Max(m,a[k]);
}
return m;
}
};
void dfs(int v,vector<vi> &g,vi <,vi &rt,int &id){
lt[v]=id++;
for(auto u:g[v]) dfs(u,g,lt,rt,id);
rt[v]=id;
}
ll Solve(int N,vi A,vi H,vi C){
vi lt(N),rt(N);
int id=0;
vector<vi> g(N);
vector<pair<int,int>> a(N);
vector<ll> b(N);
Segment_Tree st(N);
ll res=0;
for(int i=0;i<N;i++){
a[i]={H[i],i};
if(i) g[A[i]].push_back(i);
res+=C[i];
}
sort(a.rbegin(),a.rend());
dfs(0,g,lt,rt,id);
for(auto p:a){
int i=p.second;
ll t=C[i];
b[i]=C[i];
st.ChangeMax(lt[i],rt[i],i);
while(t){
int pr=(i?st.Access(lt[A[i]]):-1);
if(pr<0){
res-=t;
break;
}
i=pr;
ll mn=min(t,b[i]);
t-=mn,b[i]-=mn;
if(b[i]) continue;
st.Erase(i);
pr=(i?st.Access(lt[A[i]]):-1);
if(pr>=0) st.ChangeMax(lt[pr],rt[pr],pr);
}
}
return res;
}
int N;
vi A,H,C;
int main(){
scanf("%d",&N);
A=H=C=vi(N);
vi deg(N),ind(N,-1);
vector<vi> g(N);
queue<int> q;
ll res=0;
for(int i=0;i<N;i++){
scanf("%d%d%d",&A[i],&H[i],&C[i]);
A[i]--;
deg[A[i]]++;
g[A[i]].push_back(i);
}
for(int i=0;i<N;i++) if(!deg[i]) q.push(i);
while(!q.empty()){
int v=q.front(),u=A[v];
q.pop();
deg[u]--;
if(!deg[u]) q.push(u);
}
for(int i=0;i<N;i++) if(ind[i]<0&°[i]){
int v=i,I=1;
vector<pair<int,int>> b;
do{
b.push_back({H[v],C[v]});
ind[v]=0;
for(auto u:g[v]) if(!deg[u]){
ind[u]=I++;
q.push(u);
}
v=A[v];
}while(v!=i);
sort(b.rbegin(),b.rend());
int N_=(int)b.size();
vi A_(N_),H_,C_;
for(int i=1;i<N_;i++) A_[i]=i-1;
for(auto p:b){
H_.push_back(p.first);
C_.push_back(p.second);
}
while(!q.empty()){
int u=q.front();
q.pop();
A_.push_back(ind[A[u]]+N_-1);
H_.push_back(H[u]);
C_.push_back(C[u]);
for(auto w:g[u]){
ind[w]=I++;
q.push(w);
}
}
N_=A_.size();
res+=Solve(N_,A_,H_,C_);
}
printf("%lld\n",res);
}
컴파일 시 표준 에러 (stderr) 메시지
worst_reporter2.cpp: In function 'int main()':
worst_reporter2.cpp:99:7: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
99 | scanf("%d",&N);
| ~~~~~^~~~~~~~~
worst_reporter2.cpp:106:8: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
106 | scanf("%d%d%d",&A[i],&H[i],&C[i]);
| ~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~~~
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |