codeforces B. Levko and Permutation 解题报告

题目链接:http://codeforces.com/problemset/problem/361/B

题目意思:有n个数,这些数的范围是[1,n],并且每个数都是不相同的。你需要构造一个排列,使得这个排列上的数与它所在位置的序号的最大公约数满足 > 1,并且这些数的个数恰好满足k个,输出这样的一个排列。

      先说明什么时候得不到这样的一个排列,就是n = k的情况。因为任何一个数x放在第1个位置的gcd(x, 1) = 1的,所以要得出这样一个排列 k 最大只能为 n-1 。而k最小为0个,这个排列是n,2,3,4,...,n-1(当然也可以是2,3,...,n,n+1等等)。那么,如果k 满足 [0, n - 1]的取值范围,这样的排列绝对存在。

     接着讨论一般情况下,如果构造出满足恰好有k个符合这样条件的排列。这里先声明一个规律,当某个数x放在与它下标也是x的位置上时(这个数当然不能为1),那么绝对是满足gcd(x, x) > 1的,最大公约数即是它自己。现在推出构造这个排列的规律。

假设n = 8, k = 2

     下标i :   1         2         3         4         5       6      7         8

     排列xi:   2         3       4         5         6         1         7         8

如果 k= 3

     下标i :   1         2         3         4         5       6      7         8

     排列xi:  2         3         4       5         1         6         7         8

     其他依此类推,可以发现前n-k-1 个数都是从比它下标多1开始的,这样保证gcd(i, xi) = 1,而第n-k的位置填1,这样加起来恰好满足gcd(i, xi)的个数等于n-k,而最后的k个数则保持与它的下标相等。

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstdlib>
 4 using namespace std;
 5 
 6 int main()
 7 {
 8     int i,  n, k;
 9     while (scanf("%d%d", &n, &k) != EOF)
10     {
11         if (k == 0)
12         {
13             printf("%d ", n);
14             for (i = 1; i < n; i++)
15                 printf("%d ", i);
16         }
17         else if (k == n)
18             printf("-1\n");
19         else
20         {
21             for (i = 1; i < n - k; i++)
22                 printf("%d ", i+1);
23             printf("1 ");
24             for (i = n-k+1; i <= n; i++)
25                 printf("%d ", i);
26             printf("\n");
27         }
28     }
29     return 0;
30 }

 

posted @ 2013-11-19 16:39  windysai  阅读(424)  评论(4编辑  收藏  举报