4563: [Haoi2016]放棋子

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 440  Solved: 285
[Submit][Status][Discuss]

Description

给你一个N*N的矩阵,每行有一个障碍,数据保证任意两个障碍不在同一行,任意两个障碍不在同一列,要求你在
这个矩阵上放N枚棋子(障碍的位置不能放棋子),要求你放N个棋子也满足每行只有一枚棋子,每列只有一枚棋子
的限制,求有多少种方案。
 

Input

第一行一个N,接下来一个N*N的矩阵。N<=200,0表示没有障碍,1表示有障碍,输入格式参考样例
 

Output

一个整数,即合法的方案数。

Sample Input

2
0 1
1 0

Sample Output

1
  这道题竟然考的是高精度,吓到我了……
  一开始没读到数据范围还以为是状压裸题,然后一看到N<=200,吓一跳,然后开始琢磨动归方程,于是乎,一开始就错了的我走上了一条不归路。
  最后实在没辙,看了一眼题解,好吧,我输了。
  这道题我们可以分析为错排问题:一共 1~n n个数,对于任意数x都不在第x个位置上有多少方案数。
  为什么这么说呢?我们可以注意到,既然每一行每一列有且只有一个障碍,那么,每一行障碍的位置对于答案没有任何实际影响,如果我们按照每一行障碍的位置对行进行排序的话就转化成了第i行的棋子不能出现在第i个位置的问题,也就是我们上面说的错排问题了。
  那么错排问题的公式是什么呢?
    f[i]=f[i-1]*(i-1)+f[i-2]*(i-1)
    原理:

      第一步,把第n个元素放在一个位置,比如位置k,一共有n-1种方法;
      第二步,放编号为k的元素,这时有两种情况:⑴把它放到位置n,那么,对于剩下的n-1个元素,由于第k个元素放到了位置n,剩下n-2个元素就有f(n-2)种方法;⑵第k    个元素不把它放到位置n,这时,对于这n-1个元素,有f(n-1)种方法。(摘自百度百科)
  还是挺好玩的。
  然后,就是传统的高精度了呗。

 #include <iostream>
#include <cstdlib>
#include <cstdio>
#include <cstring>
#include <queue>
#include <algorithm>
#include <cmath>
#include <map>
#define N 10000
using namespace std;
int n,p=;
struct no
{
int a[N],l;
}f[],c;
no get(int x)
{
no aa;
aa.l=;
aa.a[]=x;
return aa;
}
no jia(no a,no b)
{
memset(c.a,,sizeof(c.a));c.l=;
for(int i=;i<=max(a.l,b.l)+;i++)
{
c.a[i]+=a.a[i]+b.a[i];
c.a[i+]+=c.a[i]/p;
c.a[i]%=p;
}
for(int i=max(a.l,b.l)+;;i--)
{
if(c.a[i])
{
c.l=i;
break;
}
}
return c;
}
no cheng(no a,no b)
{
memset(c.a,,sizeof(c.a));c.l=;
for(int i=;i<=a.l;i++)
{
for(int j=,to=i;j<=b.l;j++,to++)
{
c.a[to]+=a.a[i]*b.a[j];
c.a[to+]+=c.a[to]/p;
c.a[to]%=p;
}
}
for(int i=a.l+b.l+;;i--)
{
if(c.a[i])
{
c.l=i;
break;
}
}
return c;
}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
for(int j=;j<=n;j++)
{
int x;
scanf("%d",&x);
}
}
f[].a[]=,f[].l=;
f[].a[]=,f[].l=;
for(int i=;i<=n;i++)
{
f[i]=cheng(get(i-),jia(f[i-],f[i-]));
}
printf("%d",f[n].a[f[n].l]);
for(int i=f[n].l-;i>=;i--)
{
printf("%04d",f[n].a[i]);
}
return ;
}

最新文章

  1. apache flink 入门
  2. QQ空间HD(6)-实现自定义的选项卡切换效果
  3. Quartz定时调度配置
  4. XPath Checker和Firebug安装与使用
  5. 从零开始学Python04作业源码:模拟ATM电子银行(仅供参考)
  6. SL410K 在Ubuntu禁用触摸板
  7. HTTP层 &mdash;&mdash; 验证
  8. poj炮兵阵地(状压)(25+10+20=55)
  9. 面试前的准备---C#知识点回顾----01
  10. [SVN服务器搭建] 在myecplise下使用的 tortoise1.9 64位 跟 subversion1.9的服务器使用
  11. 学习笔记 - 用js判断页面是否加载完成实现代码
  12. Hibernate Error: a different object with the same identifier value was already associated with the session
  13. 论文笔记(2):Deep Crisp Boundaries: From Boundaries to Higher-level Tasks
  14. CSS弹性盒模型(flex box)
  15. Day07 (黑客成长日记) 函数的参数及作用
  16. vue this.$router.push和this.$route.path的区别
  17. Hadoop记录-NameNode优化
  18. Jmeter+Ant+Jenkins 接口自动化之简单demo
  19. CountDownLatch两种用法
  20. box-shadow的动效制作

热门文章

  1. 自动启动 Windows 10 UWP 应用
  2. WPF里DataGrid分页控件
  3. 第一个kotlin程序
  4. Win10版《芒果TV》获评2016年度Windows Store最佳官方/休闲娱乐应用(LiveSino和微软信仰中心联合评选)
  5. 数据可视化图表ECharts
  6. Oracle序列使用:建立、删除、使用
  7. 卸载win10内置windows app的方法
  8. chrome 浏览器的常用命令收录
  9. Java 几个有用的命令 - All Options, Memory Options, GC Options, System Properties, Thread Dump, Heap Dump
  10. PHP 的异步并行 C 扩展 Swoole