hihoCoder 1041 国庆出游 最详细的解题报告

题目来源:国庆出游

解题思路(下面是大神的写的):

把题目中的序列称作S,树称作T。那么对于S中的任意节点x,x的子孙节点如果在S出现的话,那么这个子孙节点的位置是有一定要求的:x的所有子孙节点在S中的位置都恰好紧跟在x的后面,没有被其他节点隔开。 设x的子孙节点是abcd,那么--xabcd--, --xbcda-- 等等是合法的,--xab-cd--, --axbcd--, --x--abcd--, 都是不合法的('-'指其他节点)。对于S中每个节点都做如上判断,如果有不合法的就输出NO,如果都合法就输出YES。

 

感谢两位大神gtdzx和aprilzc,解题思路是gtdzx大神提供的,代码参考aprilzc大神的

 

具体算法(java版,可以直接AC)

 1 import java.util.Scanner;
 2 
 3 public class Main {
 4     //判断child是否为father的子孙节点
 5     public static boolean isAncestor(int[] tree, int father, int child) {
 6         while (tree[child] != child) {
 7             child = tree[child];//将child的父节点赋值给child
 8             if (father == child)
 9                 return true;
10         }
11         return false;
12     }
13 
14     public static boolean judge(int[] tree, int[] array, int start, int end) {
15         if (start >= end)//所有节点都检查完了
16             return true;
17         int k = start + 1;//当前节点的下一个节点
18         //依次计算当前节点start后面的节点是否为它的子孙节点(这些节点必须是连续的)
19         while (k <= end && isAncestor(tree, array[start], array[k])) {
20             k++;
21         }
22         //如果后面所有的节点均是start的子孙节点
23         //说明该节点的已经通过检查,然后判断下一个当前节点(start+1)
24         if (k >= end) {
25             return judge(tree, array, start + 1, end);
26         }
27         int v = k + 1;
28         //判断后面的节点(除连续子孙节点外,k是第一个非连续子孙节点)
29         //是否为start的子孙节点,如果是,说明存在一个非连续的子孙节点,
30         //直接return false
31         while (v <= end) {
32             if (!isAncestor(tree, array[start], array[v])) {
33                 return false;
34             }
35             v++;
36         }
37         
38         //start已经检查完了,用来判断后面的节点(start+1)
39         return judge(tree, array, start+1,end);
40         
41         //大神的最后一句代码如下:
42         //return judge(tree, array, start, k - 1) && judge(tree, array, k, end);
43         //个人觉得效果是一样的
44     }
45 
46     public static void main(String[] args) {
47         Scanner scanner = new Scanner(System.in);
48         int T = scanner.nextInt();
49         for (int i = 0; i < T; i++) {
50             int n = scanner.nextInt();
51             int[] tree = new int[n + 1];
52             tree[1] = 1;
53             for (int j = 0; j < n - 1; j++) {
54                 int father = scanner.nextInt();
55                 int child = scanner.nextInt();
56                 tree[child] = father;
57             }
58             int m = scanner.nextInt();
59             int[] array = new int[m];
60             for (int j = 0; j < m; j++) {
61                 array[j] = scanner.nextInt();
62             }
63             if (judge(tree, array, 0, array.length - 1)) {
64                 System.out.println("YES");
65             } else {
66                 System.out.println("NO");
67             }
68         }
69         scanner.close();
70     }
71 }

 

你可能感兴趣的:(hihoCoder 1041 国庆出游 最详细的解题报告)