# | 제출 시각 | 아이디 | 문제 | 언어 | 결과 | 실행 시간 | 메모리 |
---|---|---|---|---|---|---|---|
211264 | Lawliet | Magic Tree (CEOI19_magictree) | C++14 | 1161 ms | 661452 KiB |
이 제출은 이전 버전의 oj.uz에서 채점하였습니다. 현재는 제출 당시와는 다른 서버에서 채점을 하기 때문에, 다시 제출하면 결과가 달라질 수도 있습니다.
#include <bits/stdc++.h>
using namespace std;
typedef long long int lli;
const int LOG = 30;
const int MAXK = 6010;
const int MAXN = 200010;
int n, m, k;
int curTime;
int cntNode;
int t[MAXN];
int w[MAXN];
int tin[MAXN];
int tout[MAXN];
int prof[MAXN];
int tab[LOG][MAXN];
int auxT[MAXN];
int auxW[MAXN];
int inv[MAXN];
int compressed[MAXN];
lli dp[MAXK][MAXK];
vector< int > adj[MAXN];
vector< int > auxTree[MAXN];
void DFSCalculate(int cur)
{
for(int i = 0 ; i < adj[cur].size() ; i++)
DFSCalculate( adj[cur][i] );
for(int j = 1 ; j <= k ; j++)
{
lli sum = 0;
if( t[cur] == j ) sum = w[cur];
for(int i = 0 ; i < adj[cur].size() ; i++)
sum += dp[ adj[cur][i] ][j];
dp[cur][j] = max( dp[cur][j - 1] , sum );
}
}
int LCA(int U, int V)
{
if( prof[U] < prof[V] ) swap( U , V );
int d = prof[U] - prof[V];
for(int k = 0 ; k < LOG ; k++)
if( d & (1 << k) ) U = tab[k][U];
if( U == V ) return U;
for(int k = LOG - 1 ; k >= 0 ; k--)
{
if( tab[k][U] != tab[k][V] )
{
U = tab[k][U];
V = tab[k][V];
}
}
return tab[0][U];
}
void DFSInit(int cur, int d)
{
prof[cur] = d;
tin[cur] = ++curTime;
for(int i = 0 ; i < auxTree[cur].size() ; i++)
DFSInit( auxTree[cur][i] , d + 1 );
tout[cur] = curTime;
}
bool cmp(int U, int V) { return tin[U] < tin[V]; }
int getNode(int U)
{
if( compressed[U] == 0 )
{
compressed[U] = ++cntNode;
inv[ cntNode ] = U;
}
return compressed[U];
}
void compressTree()
{
for(int k = 1 ; k <= 3 ; k++)
{
for(int i = 1 ; i <= n ; i++)
{
int jump = tab[k - 1][i];
tab[k][i] = tab[k - 1][jump];
}
}
DFSInit( 1 , 0 );
vector< int > v;
for(int i = 1 ; i <= n ; i++)
if( auxW[i] != 0 ) v.push_back( i );
v.push_back( 1 );
sort( v.begin() , v.end() , cmp );
int sz = v.size();
for(int i = 0 ; i < sz - 1 ; i++)
v.push_back( LCA( v[i] , v[i + 1] ) );
sort( v.begin() , v.end() , cmp );
vector< int > s;
s.push_back( v[0] );
for(int i = 1 ; i < v.size() ; i++)
{
if( v[i] == v[i - 1] ) continue;
while( true )
{
int last = s.back();
if( tin[last] <= tin[ v[i] ] && tout[ v[i] ] <= tout[last] ) break;
s.pop_back();
}
int U = getNode( s.back() );
int V = getNode( v[i] );
adj[ U ].push_back( V );
//printf("..... %d %d %d %d\n",U,V,inv[U],inv[V]);
s.push_back( v[i] );
}
for(int i = 1 ; i <= n ; i++)
{
if( compressed[i] == 0 ) continue;
t[ compressed[i] ] = auxT[i];
w[ compressed[i] ] = auxW[i];
}
n = cntNode;
}
void compressTime()
{
set< int > s;
map< int , int > conv;
for(int i = 1 ; i <= n ; i++)
s.insert( t[i] );
int cnt = 0;
for(auto it = s.begin() ; it != s.end() ; it++)
conv[ *it ] = ++cnt;
for(int i = 1 ; i <= n ; i++)
t[i] = conv[ t[i] ];
k = 0;
for(int i = 1 ; i <= n ; i++)
k = max( k , t[i] );
}
int main()
{
scanf("%d %d %d",&n,&m,&k);
for(int i = 2 ; i <= n ; i++)
{
scanf("%d",&tab[0][i]);
auxTree[ tab[0][i] ].push_back( i );
}
for(int i = 1 ; i <= m ; i++)
{
int node;
scanf("%d",&node);
scanf("%d %d",&auxT[node],&auxW[node]);
}
compressTree();
compressTime();
DFSCalculate( 1 );
printf("%lld\n",dp[1][k]);
}
컴파일 시 표준 에러 (stderr) 메시지
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |