Loading

Codeforces 755E:PolandBall and White-Red graph(构造+思维)

http://codeforces.com/contest/755/problem/E

题意:给出n个点和一个距离d,让你在这个n个点的图里面构造一个子图,使得这个子图的直径和补图的直径的较小值为d,如果不可能输出-1,如果可能把子图的边表示出来。

思路:一个图的直径就是图的任意两点之间的距离的最大值。

这里有参考:http://blog.csdn.net/jasonvictoryan/article/details/54572646

感觉这种类型的构造题目不会做的话还是只能找规律,当然这个规律实在挺难的。

其实我们可以发现,要么保证子图的直径 = k或者保证补图的直径 = k即可。

按照这个思路我们可以画多几个图发现:

只有 k = 2 或者 3 的时候才有解。

1、当 k = 2 时,通过样例可以发现,连一条从 1 到 n 的链即可,这样可以保证补图的直径是2。但是当 n <= 4 的时候无解。

2、当 k = 3 时,将 1 和 2, 2 和 3 连边,再将 4 到 n 的点连向 3,这样可以保证子图的直径是3。但是 n <= 3 的时候无解。

 1 #include <cstdio>
 2 #include <algorithm>
 3 #include <iostream>
 4 #include <cstring>
 5 #include <string>
 6 #include <cmath>
 7 #include <queue>
 8 #include <vector>
 9 #include <map>
10 #include <set>
11 #include <stack>
12 using namespace std;
13 #define INF 0x3f3f3f3f
14 #define N 100010
15 typedef long long LL;
16 
17 int main() {
18     int n, k;
19     cin >> n >> k;
20     if(k != 2 && k != 3) puts("-1");
21     else {
22         if(k == 2) {
23             if(n <= 4) puts("-1");
24             else {
25                 printf("%d\n", n - 1);
26                 for(int i = 1; i < n; i++) printf("%d %d\n", i, i + 1);
27             }
28         } else {
29             if(n <= 3) puts("-1");
30             else {
31                 printf("%d\n", n - 1);
32                 printf("1 2\n2 3\n");
33                 for(int i = 4; i <= n; i++) printf("3 %d\n", i);
34             }
35         }
36     }
37     return 0;
38 }

 

posted @ 2017-01-17 11:02  Shadowdsp  阅读(286)  评论(0编辑  收藏  举报