【BZOJ3120】Line

Description

Wayne喜欢排队……不对,是Wayne所在学校的校长喜欢看大家排队,尤其是在操场上站方阵。
某日课间操时,校长童心大发想了一个极具观赏性的列队方案,如下:
1. 方阵排成N行,每行恰好M个学生。
2. 由于校长喜欢女孩子,所以在一行上不能有连续P个男生。
3. 由于校长喜欢女孩子,所以在校长看来,一列全是男生是不好的,全男生的列数不能超过Q。
Wayne因为感冒了所以不用参加列队,不过他看着大家排队排得不亦乐乎,于是他想知道,在男女生数目无限制的情况下,有多少种列队方案?
两种方案被视作不同,表明存在至少一个二元组(i,j)而两种方案中第i行第j列的同学性别不同。另外,因为答案可能很大,所以请把答案模10^9 + 7。

Input

输入仅一行4个正整数,依次是N,M,P,Q。

Output

输出仅一行,表示答案。

Sample Input

2 3 3 1

Sample Output

46

HINT

【数据规模和约定】
对于5%的数据,满足P = 1。
对于另外10%的数据,满足N * M <= 20。
对于另外15%的数据,满足N <= 2,M <= 10^6。
对于另外10%的数据,满足N <= 2。
对于另外20%的数据,满足N <= 4,P <= 2,Q <= 2。
对于100%的数据,满足1 <= N <= 8,1 <= M <= 10^18,1 <= P <= 3,0 <= Q <= 3。

题解:刚看到题,第一反应是状压+矩乘,不过算了算矩阵的大小有点难以接受。不过读了读题,发现行与行之间是互不影响的,所以我们只需要知道每一列男生女生的个数即可。

用f[i][a][b][c]表示第i列有a行有连续0个男生,b行有连续1个男生,n-a-b行有连续2个男生,已经有c行全是男生的方案数,然后把后面那3维压一压变成矩阵就可以用矩乘来优化了。

注意:a+b>n的情况显然不合法,把它扔掉可以压缩矩阵大小;矩乘时特判如果该位置为0,则不进行计算,居然会快很多。

#include <cstdio>
#include <iostream>
#include <cstring>
using namespace std;
typedef long long ll;
const int P=1000000007;
int n,p,q,tot;
ll m,ans;
struct M
{
int v[200][200];
M () {memset(v,0,sizeof(v));}
int * operator [] (int a) {return v[a];}
inline M operator * (const M &b) const
{
M c;
register int i,j,k;
for(i=1;i<=tot;i++) for(k=1;k<=tot;k++) if(v[i][k])
for(j=1;j<=tot;j++) if(b.v[k][j])
c[i][j]=(c[i][j]+(ll)v[i][k]*b.v[k][j])%P;
return c;
}
}S,T;
ll C[10][10];
int _[10][10][5],__[10][5];
//a0 b1 c2
inline void pm(ll y)
{
while(y)
{
if(y&1) S=S*T;
T=T*T,y>>=1;
}
}
int main()
{
scanf("%d%lld%d%d",&n,&m,&p,&q);
int i,j,a,b,a1,b1,x,y;
for(i=0;i<=n;i++)
{
C[i][0]=1;
for(j=1;j<=i;j++) C[i][j]=(C[i-1][j-1]+C[i-1][j])%P;
}
if(p==1)
{
printf("1");
return 0;
}
if(p==2)
{
for(a=0;a<=n;a++) for(b=0;b<=q;b++) __[a][b]=++tot;
for(a=0;a<=n;a++) for(a1=0;a1<=a;a1++)
{
x=n-a1;
if(a1<n) for(i=0;i<=q;i++) T[__[a][i]][__[x][i]]=C[a][a1];
else for(i=0;i<q;i++) T[__[a][i]][__[x][i+1]]=C[a][a1];
}
S[1][__[n][0]]=1;
pm(m);
for(i=1;i<=tot;i++) ans=(ans+S[1][i])%P;
printf("%lld",ans);
return 0;
}
for(a=0;a<=n;a++) for(b=0;a+b<=n;b++) for(i=0;i<=q;i++) _[a][b][i]=++tot;
for(a=0;a<=n;a++) for(b=0;a+b<=n;b++)
{
for(a1=0;a1<=a;a1++) for(b1=0;b1<=b;b1++)
{
x=n-a1-b1,y=a1;
if(a1+b1<n) for(i=0;i<=q;i++) T[_[a][b][i]][_[x][y][i]]=C[a][a1]*C[b][b1]%P;
else for(i=0;i<q;i++) T[_[a][b][i]][_[x][y][i+1]]=C[a][a1]*C[b][b1]%P;
}
}
S[1][_[n][0][0]]=1;
pm(m);
for(i=1;i<=tot;i++) ans=(ans+S[1][i])%P;
printf("%lld",ans);
return 0;
}//8 1000000000000000000 3 3

【BZOJ3120】Line 矩阵乘法的更多相关文章

  1. *HDU 1757 矩阵乘法

    A Simple Math Problem Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Ot ...

  2. POJ3070 Fibonacci[矩阵乘法]

    Fibonacci Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 13677   Accepted: 9697 Descri ...

  3. 【Codeforces718C】Sasha and Array 线段树 + 矩阵乘法

    C. Sasha and Array time limit per test:5 seconds memory limit per test:256 megabytes input:standard ...

  4. 矩阵乘法的MapReduce实现

    对于任意矩阵M和N,若矩阵M的列数等于矩阵N的行数,则记M和N的乘积为P=M*N,其中mik 记做矩阵M的第i行和第k列,nkj记做矩阵N的第k行和第j列,则矩阵P中,第i行第j列的元素可表示为公式( ...

  5. hdu4920 Matrix multiplication 模3矩阵乘法

    hdu4920 Matrix multiplication Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 131072/131072 ...

  6. 矩阵乘法快速幂 codevs 1250 Fibonacci数列

    codevs 1250 Fibonacci数列  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond   题目描述 Description 定义:f0=f1=1 ...

  7. POJ3613 Cow Relays [矩阵乘法 floyd类似]

    Cow Relays Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7335   Accepted: 2878 Descri ...

  8. 【POJ2778】AC自动机+矩阵乘法

    DNA Sequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14758 Accepted: 5716 Descrip ...

  9. 数学(矩阵乘法):HDU 4565 So Easy!

    So Easy! Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Su ...

随机推荐

  1. 三维模型 DAE 导出格式结合 OpenGLES 要素浅析

    三维模型 DAE 导出格式结合 OpenGLES 要素浅析 太阳火神的漂亮人生 (http://blog.csdn.net/opengl_es) 本文遵循"署名-非商业用途-保持一致&quo ...

  2. Mycat和MySQL的差别——Mycat的核心作用

    有个朋友面试的时候被问到:Mycat和MySQL的差别.我们能够把上层看作是对下层的抽象,比如操作系统是对各类计算机硬件的抽象.那么我们什么时候须要抽象?假如仅仅有一种硬件的时候,我们须要开发一个操作 ...

  3. LINQ to SQL语句之Select/Distinct和Count/Sum/Min/Max/Avg (转)

    Select/Distinct操作符 适用场景:o(∩_∩)o… 查询呗. 说明:和SQL命令中的select作用相似但位置不同,查询表达式中的select及所接子句是放在表达式最后并把子句中的变量也 ...

  4. blender, 创建多边形面片

    按a键清除所有选择,进入Edit Mode,选vertex select方式.然后按住control,使用MLB连续画多个顶点,形成一个多边形,如图所示: 然后同时选中两个端点,点Make Edge/ ...

  5. Atitit. Dwr 抛出异常error解决方案

    Atitit. Dwr 抛出异常error解决方案 1. Dwr3的处理机制..setErrorHandler 1 2. remote Mteh  try catch 1 3. 林吧,子好java 处 ...

  6. Ecshop安装的坑,建议不要使用!

    最近因为工作的需要,安装了下ecshop,这个曾经的火爆开源程序,现在也呈现出疲态. 1.请看官方的运行环境推荐: 服务器端运行环境推荐·php版本5.0以上5.3以下的版本(推荐使用5.2系列版本) ...

  7. oracle中select clob的返回类型

    当select的字段是clob类型的数据时,但是数据长度在2000字节到4000字节时,默认转为long类型. 所以当用insert into select的时候,预期插入的是clob类型,但是报or ...

  8. systemd启动多实例

    最近用了centos7,启动管理器用的是systemd,感觉很好玩. 1.开机自动启动 新建一个service文件放到/usr/lib/systemd/system/ 比如: [Unit] Descr ...

  9. lua工具库penlight--09技术选择

    模块化和粒度 在理想的世界,一个程序应该只加载它需要的库.Penlight需要额外100 Kb 的字节码来工作.它是简单但却乏味要加载你需要什么: local data = require 'pl.d ...

  10. 编写自己的代码库(javascript常用实例的实现与封装)

    https://segmentfault.com/a/1190000010225928