试题 算法训练 摆动序列
资源限制
时间限制:1.0s 内存限制:512.0MB
问题描述
如果一个序列满足下面的性质,我们就将它称为摆动序列:
1. 序列中的所有数都是不大于k的正整数;
2. 序列中至少有两个数。
3. 序列中的数两两不相等;
4. 如果第i – 1个数比第i – 2个数大,则第i个数比第i – 2个数小;如果第i – 1个数比第i – 2个数小,则第i个数比第i – 2个数大。
比如,当k = 3时,有下面几个这样的序列:
1 2
1 3
2 1
2 1 3
2 3
2 3 1
3 1
3 2
一共有8种,给定k,请求出满足上面要求的序列的个数。
输入格式
输入包含了一个整数k。(k<=20)
输出格式
输出一个整数,表示满足要求的序列个数。
样例输入
3
样例输出
8
方法一 ( 数学方法 )
基本思路::
根据条件 我们能得到一些信息(需要思考一下)
1. 只要给出一个大于等于2的数组,都能形成摆动序列
2. 能够形成的摆动序列中,又根据先增或先减分出2种情况(一种逐渐发散 一种逐渐收敛)
3. 长度为n的数组 能形成(无序) C(n,1) 个长度为1子数组 形成 C(n,2) 个长度为 2的子数组......形成长度为 C(n,n)长度为n的数组
4. 又因为长度为1的不算是摆动序列 因此无序的情况下 能排列出来的数组个数是 C(n,0) + C(n,1) + C(n,2) + C(n,3) + C(n,4) +C(n,5) + C(n,6) +C(n,7) + C(n,8) + ...... +C(n,n) - C(n,1) - C(n,0) = 2^n - n - 1
5. 由于拥有2中情况 那么 最后 答案 就是 (2^n - n - 1) * 2
代码实现::
import java.io.*;
import java.util.StringTokenizer;
class Main {
ReadIn readIn = new ReadIn(System.in);
PrintWriter printWriter = new PrintWriter(System.out,true);
public void run() throws IOException {
int n = readIn.nextInt();
printWriter.println(((int)Math.pow(2,n) - n - 1) << 1);
}
public