CF 1025 D. Recovering BST
D. Recovering BST
http://codeforces.com/contest/1025/problem/D
题意:
给出一个连续上升的序列a,两个点之间有边满足gcd(ai ,aj) != 1。选择一些边,问是否能构成一棵有n个点的二叉搜索树。
分析:
区间dp。
每个子树都是一段连续的区间,L[i][j]表示j为根,i~j-1这个区间的点能否使j的左子树,R[i][j]:i为根,i+1~j这个区间能否为i的右子树。
枚举一个中间点作为根,转移即可。
为什么这样转移:直接f[i][j]表示区间i~j能否构成一个树的话,还要去枚举根,或者记录根。就变成了f[i][j][k]表示区间i~j,根为k能否成为一棵树。这样复杂度就太大了。换一种记录根的方式,只记录左边和右边,两边是互不影响的,前面的状态拆成了两个,现在就不需要枚举根了。
代码:
1 #include<cstdio> 2 #include<algorithm> 3 #include<cstring> 4 #include<cmath> 5 #include<iostream> 6 #include<cctype> 7 #include<set> 8 #include<vector> 9 #include<queue> 10 #include<map> 11 #define fi(s) freopen(s,"r",stdin); 12 #define fo(s) freopen(s,"w",stdout); 13 using namespace std; 14 typedef long long LL; 15 16 inline int read() { 17 int x=0,f=1;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-1; 18 for(;isdigit(ch);ch=getchar())x=x*10+ch-'0';return x*f; 19 } 20 21 const int N = 705; 22 23 LL a[N]; 24 bool L[N][N], R[N][N], e[N][N]; 25 26 LL gcd(LL a,LL b) { 27 return b == 0 ? a : gcd(b, a % b); 28 } 29 30 int main() { 31 32 int n = read(); 33 for (int i = 1; i <= n; ++i) { 34 a[i] = read(); 35 L[i][i] = R[i][i] = 1; 36 } 37 for (int i = 1; i <= n; ++i) 38 for (int j = i + 1; j <= n; ++j) 39 if (gcd(a[i], a[j]) > 1) e[i][j] = e[j][i] = 1; 40 41 for (int k = 1; k <= n; ++k) { 42 for (int i = 1, j; (j = i + k - 1) <= n; ++i) { 43 for (int mid = i; mid <= j; ++mid) { 44 if (L[i][mid] && R[mid][j]) 45 R[i - 1][j] |= e[i - 1][mid], L[i][j + 1] |= e[mid][j + 1]; 46 } 47 } 48 } 49 50 51 for (int i = 1; i <= n; ++i) { 52 if (L[1][i] && R[i][n]) { 53 puts("Yes"); return 0; 54 } 55 } 56 puts("No"); 57 return 0; 58 }