目录

CF1463E-Plan of Lectures

CF1463E-Plan of Lectures

题目:

题目描述:

Ivan is a programming teacher. During the academic year, he plans to give $ n $ lectures on $ n $ different topics. Each topic should be used in exactly one lecture. Ivan wants to choose which topic will he explain during the $ 1 $ -st, $ 2 $ -nd, …, $ n $ -th lecture — formally, he wants to choose some permutation of integers from $ 1 $ to $ n $ (let’s call this permutation $ q $ ). $ q_i $ is the index of the topic Ivan will explain during the $ i $ -th lecture.

For each topic (except exactly one), there exists a prerequisite topic (for the topic $ i $ , the prerequisite topic is $ p_i $ ). Ivan cannot give a lecture on a topic before giving a lecture on its prerequisite topic. There exists at least one valid ordering of topics according to these prerequisite constraints.

Ordering the topics correctly can help students understand the lectures better. Ivan has $ k $ special pairs of topics $ (x_i, y_i) $ such that he knows that the students will understand the $ y_i $ -th topic better if the lecture on it is conducted right after the lecture on the $ x_i $ -th topic. Ivan wants to satisfy the constraints on every such pair, that is, for every $ i \in [1, k] $ , there should exist some $ j \in [1, n - 1] $ such that $ q_j = x_i $ and $ q_{j + 1} = y_i $ .

Now Ivan wants to know if there exists an ordering of topics that satisfies all these constraints, and if at least one exists, find any of them.

输入格式:

The first line contains two integers $ n $ and $ k $ ( $ 2 \le n \le 3 \cdot 10^5 $ , $ 1 \le k \le n - 1 $ ) — the number of topics and the number of special pairs of topics, respectively.

The second line contains $ n $ integers $ p_1 $ , $ p_2 $ , …, $ p_n $ ( $ 0 \le p_i \le n $ ), where $ p_i $ is the prerequisite topic for the topic $ i $ (or $ p_i = 0 $ if the $ i $ -th topic has no prerequisite topics). Exactly one of these integers is $ 0 $ . At least one ordering of topics such that for every $ i $ the $ p_i $ -th topic is placed before the $ i $ -th topic exists.

Then $ k $ lines follow, the $ i $ -th line contains two integers $ x_i $ and $ y_i $ ( $ 1 \le x_i, y_i \le n $ ; $ x_i \ne y_i $ ) — the topics from the $ i $ -th special pair. All values of $ x_i $ are pairwise distinct; similarly, all valus of $ y_i $ are pairwise distinct.

输出格式:

If there is no ordering of topics meeting all the constraints, print $ 0 $ .

Otherwise, print $ n $ pairwise distinct integers $ q_1 $ , $ q_2 $ , …, $ q_n $ ( $ 1 \le q_i \le n $ ) — the ordering of topics meeting all of the constraints. If there are multiple answers, print any of them.

样例:

样例输入1:

1
2
3
4
5 2
2 3 0 5 3
1 5
5 4

样例输出1:

1
3 2 1 5 4

样例输入2:

1
2
3
4
5 2
2 3 0 5 3
1 5
5 1

样例输出2:

1
0

样例输入3:

1
2
3
5 1
2 3 0 5 3
4 5

样例输出3:

1
0

样例输入4:

1
2
3
4
5
6
5 4
2 3 0 5 3
2 1
3 5
5 2
1 4

样例输出4:

1
3 5 2 1 4

思路:

实现:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
#include "ybwhead/ios.h"
const int maxn = 300005;

int p[maxn];
int find(int x)
{
    return p[x] == x ? x : p[x] = find(p[x]);
}
int a[maxn], b[maxn], c[maxn], in[maxn], ans[maxn];
vector<int> G[maxn], H[maxn];
bool vis[maxn];
int main()
{
    int n, m;
    yin >> n >> m;
    int root;
    for (int i = 1; i <= n; i++)
    {
        int fa;
        yin >> fa;
        p[i] = i;
        if (fa == 0)
            root = i;
        else
            H[fa].push_back(i);
    }
    while (m--)
    {
        int x, y;
        yin >> x >> y;
        a[x] = y;
        b[y] = x;
        x = find(x), y = find(y);
        if (x == y)
            return puts("0"), 0;
        p[x] = y;
    }
    for (int i = 1; i <= n; i++)
    {
        if (b[i] == 0)
            c[find(i)] = i; // 起点
        for (auto v : H[i])
        {
            if (find(i) != find(v))
            {
                G[find(i)].push_back(find(v));
                in[find(v)]++;
            }
        }
    }
    queue<int> q;
    q.push(find(root));
    int ct = 0;
    while (q.size())
    {
        int u = q.front();
        q.pop();
        int t = c[u];
        while (t)
        {
            ans[++ct] = t;
            t = a[t];
        }
        for (auto v : G[u])
        {
            in[v]--;
            if (in[v] == 0)
            {
                q.push(v);
            }
        }
    }
    if (ct != n)
        return puts("0"), 0;
    vis[root] = 1;
    for (int i = 1; i <= n; i++)
    {
        if (vis[ans[i]] == 0)
            return puts("0"), 0;
        for (auto v : H[ans[i]])
            vis[v] = 1;
    }
    for (int i = 1; i <= n; i++)
        yout << ans[i] << " ";
    return 0;
}