이 제출은 이전 버전의 oj.uz에서 채점하였습니다. 현재는 제출 당시와는 다른 서버에서 채점을 하기 때문에, 다시 제출하면 결과가 달라질 수도 있습니다.
#include "werewolf.h"
//#include "grader.cpp"
#include <vector>
#include <assert.h>
#include <iostream>
#include <functional>
using namespace std;
const int MAXN = 2e5 + 5;
const int MAXLog = 18;
int n;
vector <int> graph[MAXN];
bool solveDP(int s, int e, int l, int r)
{
if(s<l || e>r) return false;
vector <bool> used[2];
used[0].assign(n+5, false);
used[1].assign(n+5, false);
function <void(int, int)> DFS = [&](int x, int state)
{
if(used[state][x]==true) return;
used[state][x] = true;
//cout << state << " " << x << '\n';
if(state==0 && x>=l && x<=r) DFS(x, 1);
for(int y: graph[x])
{
if(state==0 && y<l) continue;
if(state==1 && y>r) continue;
DFS(y, state);
}
};
DFS(s, 0);
return used[1][e];
}
struct Element
{
int minVal, maxVal;
Element(){}
Element(int x)
{
this->minVal = x;
this->maxVal = x;
}
};
Element Merge(Element A, Element B)
{
Element out;
out.minVal = min(A.minVal, B.minVal);
out.maxVal = max(A.maxVal, B.maxVal);
return out;
}
int orderPos[MAXN];
vector <int> dfsOrder;
int logVal[MAXN];
Element sparse[20][MAXN];
void initLine()
{
int root = -1;
for(int x = 0;x<n;x++)
{
if(graph[x].size()==1)
{
root = x;
break;
}
}
function <void(int, int)> DFS = [&](int x, int last)
{
orderPos[x] = dfsOrder.size();
dfsOrder.push_back(x);
for(int y: graph[x])
{
if(y==last) continue;
DFS(y, x);
}
};
DFS(root, -1);
for(int i = 0;i<n;i++) sparse[0][i] = Element(dfsOrder[i]);
for(int i = 1;i<=MAXLog;i++)
{
for(int j = 0;j<n;j++)
{
if(j+(1<<(i-1))<n)
sparse[i][j] = Merge(sparse[i-1][j], sparse[i-1][j+(1<<(i-1))]);
else
sparse[i][j] = sparse[i-1][j];
}
}
logVal[0] = logVal[1] = 0;
for(int i = 2;i<=n;i++) logVal[i] = logVal[i/2] + 1;
}
Element getSegment(int l, int r)
{
int log2 = logVal[r-l+1];
return Merge(sparse[log2][l], sparse[log2][r-(1<<log2)+1]);
}
bool solveLine1(int S, int E, int L, int R)
{
if(S<L || E>R) return false;
int sPos = orderPos[S];
int ePos = orderPos[E];
int ind1, ind2;
assert(sPos<ePos);
int l = sPos, r = ePos, mid;
while(l+1<r)
{
mid = (l+r)/2;
if(getSegment(sPos, mid).minVal>=L) l = mid;
else r = mid - 1;
}
if(getSegment(sPos, r).minVal>=L) ind1 = r;
else if(getSegment(sPos, l).minVal>=L) ind1 = l;
l = sPos, r = ePos, mid;
while(l+1<r)
{
mid = (l+r)/2;
if(getSegment(mid, ePos).maxVal<=R) r = mid;
else l = mid + 1;
}
if(getSegment(l, ePos).maxVal<=R) ind2 = l;
else if(getSegment(r, ePos).maxVal<=R) ind2 = r;
//cout << ind1 << " " << ind2 << '\n';
return (ind1>=ind2);
}
bool solveLine2(int S, int E, int L, int R)
{
swap(S, E);
if(S>R || E<L) return false;
int sPos = orderPos[S];
int ePos = orderPos[E];
int ind1, ind2;
int l = sPos, r = ePos, mid;
while(l+1<r)
{
mid = (l+r)/2;
if(getSegment(sPos, mid).maxVal<=R) l = mid;
else r = mid - 1;
}
if(getSegment(sPos, r).maxVal<=R) ind1 = r;
else if(getSegment(sPos, l).maxVal<=R) ind1 = l;
l = sPos, r = ePos, mid;
while(l+1<r)
{
mid = (l+r)/2;
if(getSegment(mid, ePos).minVal>=L) r = mid;
else l = mid + 1;
}
if(getSegment(l, ePos).minVal>=L) ind2 = l;
else ind2 = r;
//cout << ind1 << " " << ind2 << '\n';
return (ind1>=ind2);
}
bool solveLine(int S, int E, int L, int R)
{
if(orderPos[S]<orderPos[E]) return solveLine1(S, E, L, R);
return solveLine2(S, E, L, R);
}
vector<int> check_validity(int N, vector<int> X, vector<int> Y,
vector<int> S, vector<int> E,
vector<int> L, vector<int> R)
{
n = N;
for(int i = 0;i<X.size();i++)
{
graph[ X[i] ].push_back(Y[i]);
graph[ Y[i] ].push_back(X[i]);
}
vector <int> queryAnswers;
if(n<=6000 && X.size()<=6000 && S.size())
{
for(int i = 0;i<S.size();i++)
{
queryAnswers.push_back(solveDP(S[i], E[i], L[i], R[i]));
}
}
else
{
initLine();
for(int i = 0;i<S.size();i++)
{
queryAnswers.push_back(solveLine(S[i], E[i], L[i], R[i]));
}
}
return queryAnswers;
}
/*
6 6 3
5 1
1 2
1 3
3 4
3 0
5 2
4 2 1 2
4 2 2 2
5 4 3 4
6 5 2
0 2
2 1
1 5
5 4
4 3
0 1 0 2
1 0 1 2
*0 2 1 5 4 3
*/
컴파일 시 표준 에러 (stderr) 메시지
werewolf.cpp: In function 'bool solveLine1(int, int, int, int)':
werewolf.cpp:141:28: warning: right operand of comma operator has no effect [-Wunused-value]
141 | l = sPos, r = ePos, mid;
| ^
werewolf.cpp: In function 'bool solveLine2(int, int, int, int)':
werewolf.cpp:176:28: warning: right operand of comma operator has no effect [-Wunused-value]
176 | l = sPos, r = ePos, mid;
| ^
werewolf.cpp: In function 'std::vector<int> check_validity(int, std::vector<int>, std::vector<int>, std::vector<int>, std::vector<int>, std::vector<int>, std::vector<int>)':
werewolf.cpp:203:20: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
203 | for(int i = 0;i<X.size();i++)
| ~^~~~~~~~~
werewolf.cpp:213:24: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
213 | for(int i = 0;i<S.size();i++)
| ~^~~~~~~~~
werewolf.cpp:221:24: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
221 | for(int i = 0;i<S.size();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... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |