codeforces B. Permutation
解题报告
题目意思:给定n和k的值,需要构造一条长度为2n(每个元素取值范围只能是[1,2n])且元素各不相同的序列,这条序列符合等式。
首先非常感谢乌冬兄和syy的帮助!!尤其是乌冬兄。
刚开始做的时候完全没有思路,写出那条等式的展开式也没有发现规律。在他们的思维引导之下,我终于明白了这个问题其实可以简化为求一对数(假设为ai-1,ai)的差,这个差 = k 即可,但是k必须为负数!!!也就是ai-1 < ai。除此,其他对数只需要满足aj-1 > aj 即可。
为什么可以这样做?注意这条等式。除了那一对数的差之外,其他对数是可以消去的,前提是这些对数的关系必须和要求的那对数的差的关系正好相反。也就是说,如果那一对数的差满足 ai-1 > ai,那么其余的对数必须是aj-1 < aj。这样我们就可以将问题解决的核心转移到求出那一对数的差中。注意等号右边是2k(k >= 0),那么|ai-1 - ai| = k,ai-1-ai = -k,这样相减 k - (-k) 刚好等于2k。
求出那一对数的差,我用了一个很简便的办法,就是假设aj-1 = 2n(序列中最大的那个数),aj = 2n - k,除了这两个数之外,其他的数都满足从大到小的顺序(也就是逆序枚举)。特别要注意k = 0的情况,此时就没有必要求出 aj-1 和 aj 了,所有数从2n ~ 1(当然也可以是1~2n)按顺序输出即可。
1 #include <iostream>
2 #include <cstdio>
3 #include <cstdlib>
4 using namespace std;
5
6 int main()
7 {
8 int n, k, i, j;
9 while (scanf("%d%d", &n, &k) != EOF)
10 {
11 if (k != 0)
12 {
13 j = 2 * n - k;
14 printf("%d %d ", j, 2 * n);
15 for (i = 2 * n - 1; i >= 1; i--)
16 {
17 if (i != j)
18 printf("%d ", i);
19 }
20 }
21 else
22 {
23 for (i = 2 * n; i >= 1; i--)
24 printf("%d ", i);
25 }
26 printf("\n");
27 }
28 return 0;