Chinaunix首页 | 论坛 | 博客
  • 博客访问: 2857121
  • 博文数量: 471
  • 博客积分: 7081
  • 博客等级: 少将
  • 技术积分: 5369
  • 用 户 组: 普通用户
  • 注册时间: 2012-01-04 21:55
文章分类

全部博文(471)

文章存档

2014年(90)

2013年(69)

2012年(312)

分类: C/C++

2012-06-10 23:32:47

递推的方法推导错排公式
  当n个编号元素放在n个编号位置,元素编号与位置编号各不对应的方法数用M(n)表示,那么M(n-1)就表示n-1个编号元素放在n-1个编号位置,各不对应的方法数,其它类推.
  第一步,把第n个元素放在一个位置,比如位置k,一共有n-1种方法;
  第二步,放编号为k的元素,这时有两种情况.1,把它放到位置n,那么,对于剩下的n-2个元素,就有M(n-2)种方法;2,不把它放到位置n,这时,对于这n-2个元素,有M(n-1)种方法;
  综上得到
  M(n)=(n-1)[M(n-2)+M(n-1)]
  特殊地,M(1)=0,M(2)=1

题目地址:

思路:

传说中的“错排公式”=。=

当有n封信时,前n-1封或前n-2封可能错。

如果前n-1封错,则可以把第n封与前面任意n-1封交换而使n封全错。有f(n-1)*(n-1)

如果前n-2封错,则可以把对的那封与第n封交换,而对的那封可以是n-1封内任意一封。所以有f(n-2)*(n-1)

所以 f(n) = (f(n-1)+f(n-2)) * (n-1)

 
 

代码:

 

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21

// Author: Tanky Woo // HDOJ 1465  

 #include

 using namespace std;  

__int64 f[22];

int main()

{  

f[1] = 0;

f[2] = 1;

f[3] = 2;

for(int i = 4; i <= 20; ++i)

f[i] = (i-1)*(f[i-2]+f[i-1]);

int num;

while(scanf("%d", &num) != EOF) //是说用__int64没效果,原来输出时忘了加%I64d

printf("%I64d\n", f[num]); return 0; }

阅读(922) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~