1 2Sample Output
50.00%
这个题实质就是求解数学数列里面的错排问题:这里补充一下错排的知识:
错排问题是组合数学中的问题之一。考虑一个有n个元素的排列,若一个排列中所有的元素都不在自己原来的位置上,那么这样的排列就称为原排列的一个错排。 n个元素的错排数记为Dn。 研究一个排列错排个数的问题,叫做错排问题或称为更列问题。
最早研究错排问题的是尼古拉·伯努利和欧拉,因此历史上也称为伯努利-欧拉的装错信封的问题。这个问题有许多具体的版本,如在写信时将n封信装到n个不同的信封里,有多少种全部装错信封的情况?又比如四人各写一张贺年卡互相赠送,有多少种赠送方法?自己写的贺年卡不能送给自己,所以也是典型的错排问题。
例如有封写好了的信,收件人不同,胡乱放入个写了地址的信封中,寄出,求没有一个收件人收到他所应接收的信的概率。当,在4! = 24个排列之中,只有9个是错排:
所以有关概率为9/24 = 37.5%
推导:
显然D1=0,D2=1。当n≥3时,不妨设n排在了第k位,其中k≠n,也就是1≤k≤n-1。那么我们现在考虑第n位的情况。
所以当n排在第k位时共有Dn-2+Dn-1种错排方法,又k有从1到n-1共n-1种取法,我们可以得到:
在上面我们得到Dn=(n-1)(Dn-1+Dn-2) 从这个公式中我们可以推出Dn的通项公式,方法如下:
为书写方便,记Dn = n!Mn,则M1 = 0, M2 =
当n大于等于3时,由Dn = (n-1)(Dn-1 + Dn-2),即。 所以,。
于是有
所以
将上面式子分边累加,得
因此,我们得到错排公式
http://blog.csdn.net/Ber_Bai/article/details/77112975
import java.util.*;
public class Main {
static Scanner in = new Scanner(System.in);
static double[] a=new double[25];
static void init(){
a[1]=0;
a[2]=1;
for (int i = 3; i <=20; i++) {
a[i]=(i-1)*(a[i-2]+a[i-1]);
}
}
public static void main(String[] args) {
init();
int k=in.nextInt(),m;
double sum=1;
String p;
while(k-->0){
int n=in.nextInt();
sum=1;
for (int i =1; i <=n; i++) {
sum*=i;
}
p=String.format("%.2f",a[n]*100/sum);
System.out.println(p+"%");
}
}
}
注意输出格式的要求: