\(大意是用数组a里的数字,组成一个序列,使得序列和为n的方案种数\)传送门

\(先考虑dp.\)

\(但是不能直接用背包转移,因为是序列,要考虑顺序。\)

\(所以,为了去重,我们令dp[i][j]为凑成i最后用的a[j]的方案数\)

dp[0]=1;//把第二维优化掉
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(i>=a[j])
dp[i]+=dp[i-a[j]];

\(接下来考虑用矩阵加速。\)

\(设a数组中最大的数是size,那么dp[i]最多从dp[i-size]转移过来,所以矩阵大小是size*size\)

现在我们想用

\[\left[
\begin{matrix}
dp_{size-1}\\
dp_{size-2}\\
dp_{size-3}\\
....\\
1\\
0\\
\end{matrix}
\right]
得到
\left[
\begin{matrix}
dp_{size}\\
dp_{size-1}\\
dp_{size-2}\\
....\\
1\\
0\\
\end{matrix}
\right]
\]

那我们的构造矩阵是怎样的呢?

\(第一行因为dp[i]可以从每一个i-a[j]得到,所以所有mat[1][a[j]]=1;\)

\(其余行,只需要把mat[i][i-1]设置成1即可(下面的构造矩阵省略了第一行,因为是根据具体数据填写)\)

\[\left[
\begin{matrix}
dp_{size-1}\\
dp_{size-2}\\
dp_{size-3}\\
....\\
1\\
0\\
\end{matrix}
\right]
*
\left[
\begin{matrix}
0&1&0&0&...&0\\
0&0&1&0&...&0\\
0&0&0&1&...&0\\
0&0&0&0&...&1\\
\end{matrix}
\right]
=
\left[
\begin{matrix}
dp_{size}\\
dp_{size-1}\\
dp_{size-2}\\
....\\
1\\
0\\
\end{matrix}
\right]
\]

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll mod=1e9+7;
ll n,pr,x,nf,a[109],dp[109],size,b[109],c[109];
struct rce{
ll m[102][102];
rce(){memset(m,0,sizeof(m));}
};
rce operator * (rce a,rce b)
{
rce c;
for(int i=1;i<=size;i++)
for(int j=1;j<=size;j++)
{
c.m[i][j]=0;
for(int k=1;k<=size;k++)
c.m[i][j]=(c.m[i][j]+a.m[i][k]*b.m[k][j]%mod)%mod;
}
return c;
}
rce quickpow(rce a,ll n)
{
rce ans;
for(int i=1;i<=size;i++)
for(int j=1;j<=size;j++)
if(i==j) ans.m[i][j]=1;
else ans.m[i][j]=0;
while(n)
{
if(n&1) ans=ans*a;
a=a*a;
n>>=1;
}
return ans;
}
int main()
{
cin>>n;
cin>>pr;
for(int i=1;i<=pr;i++)
{
cin>>x;
a[x]=1;
}
cin>>nf;
for(int i=1;i<=nf;i++)
{
cin>>x;
c[x]=1;
}
for(ll i=1;i<=100;i++)
{
if(a[i]&&c[i])
{
b[++b[0]]=i;
size=max(size,i);
}
}
dp[0]=1;
for(int i=1;i<=size;i++)
for(int j=1;j<=b[0];j++)
{
if(i>=b[j])
dp[i]=(dp[i]+dp[i-b[j]])%mod;
}
rce zao,init;
for(int i=1;i<=size;i++) init.m[i][1]=dp[size-i];
for(int i=1;i<=b[0];i++) zao.m[1][b[i]]=1;
for(int i=2;i<=size;i++) zao.m[i][i-1]=1;
zao=quickpow(zao,n-size+1)*init;
cout<<zao.m[1][1]%mod;
}

P5343 【XR-1】分块(dp矩阵加速)的更多相关文章

  1. HDU 5564 Clarke and digits 状压dp+矩阵加速

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5564 题意: 求长度在[L,R]范围,并且能整除7的整数的总数. 题解: 考虑最原始的想法: dp[ ...

  2. Codeforces Round #341 (Div. 2) E. Wet Shark and Blocks dp+矩阵加速

    题目链接: http://codeforces.com/problemset/problem/621/E E. Wet Shark and Blocks time limit per test2 se ...

  3. bzoj 4037: [HAOI2015]数字串拆分【dp+矩阵加速】

    首先f长得就很像能矩阵优化的,先构造转移矩阵(这里有一点神奇的地方,我看网上的blog和我构造的矩阵完全不一样还以为我的构造能力又丧失了,后来惊奇的发现我把那篇blog里的构造矩阵部分换成我的构造方式 ...

  4. bzoj 1009 DP+矩阵加速

    我们用DP来解决这个问题 W[I,J]表示准考证的第I位,和不吉利的数匹配到了第J位的方案数,这个状态的表示也可以看成 当前到第I位了,准考证的后J位是不吉利的数的前J位,的方案数 那么我们最后的an ...

  5. bzoj2004公交线路——DP+矩阵加速递推

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2004 求方案数,想到DP: 因为两个站间距离<=p,所以每p个站中所有车一定都会停靠至 ...

  6. [Bzoj2004][Hnoi2010]Bus 公交线路(状压dp&&矩阵加速)

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=2004 看了很多大佬的博客才理解了这道题,菜到安详QAQ 在不考虑优化的情况下,先推$dp ...

  7. 洛谷P3502 [POI2010]CHO-Hamsters感想及题解(图论+字符串+矩阵加速$dp\&Floyd$)

    洛谷P3502 [POI2010]CHO-Hamsters感想及题解(图论+字符串+矩阵加速\(dp\&Floyd\)) 标签:题解 阅读体验:https://zybuluo.com/Junl ...

  8. [BZOJ 4818/LuoguP3702][SDOI2017] 序列计数 (矩阵加速DP)

    题面: 传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=4818 Solution 看到这道题,我们不妨先考虑一下20分怎么搞 想到暴力,本蒟 ...

  9. Codeforces 691E Xor-sequences(矩阵加速DP)

    题目链接 Xor-sequences 利用矩阵加速. 先预处理出当序列长度为$2$的时候的方案数. 也就是说这个序列起点是$a[i]$终点是$a[j]$且中间没有任何元素. 但是所求的$k$很大,序列 ...

随机推荐

  1. 10-jmeter 测试soap协议v1.2版本请求

    1.因为jmeter安装了第三方插件jmeter-plugins-manager.jar之后(具体安装看之前文章),此时就可简单直接测试soap协议1.2版本的请求了 2. 3.进行运行线程就可实现了 ...

  2. Powershell如何制定属性并输出

    这个标题看着有些云里雾里.... 前一阵,群里有个朋友问博主“我想把所有用户的SMTP地址全部输出到CSV文件中进行统计,但是SMTP地址似乎输出的是错误的,可在shell里看输出的内容是正确的阿” ...

  3. CountDownLatch 计算器(具有回调功能)

    final CountDownLatch cdl = new CountDownLatch(1); new Thread(new Runnable() { @Override public void ...

  4. [科普向] Roguelike游戏到底是什么?

    简单的说 Roguelike 是 RPG(角色扮演游戏)的一个分支,也是最重要的一个分支.这个名字源于 1980 年发布的著名电子游戏<Rogue>.按字面上理解,Roguelike 就是 ...

  5. 数论-质因数(gcd) UVa 10791 - Minimum Sum LCM

    https://vjudge.net/problem/UVA-10791/origin 以上为题目来源Google翻译得到的题意: 一组整数的LCM(最小公倍数)定义为最小数,即 该集合的所有整数的倍 ...

  6. Python冒泡排序算法及其优化

    冒泡排序 所谓冒泡,就是将元素两两之间进行比较,谁大就往后移动,直到将最大的元素排到最后面,接着再循环一趟,从头开始进行两两比较,而上一趟已经排好的那个元素就不用进行比较了.(图中排好序的元素标记为黄 ...

  7. 14.移动端图片浏览组件 react-wx-images-viewer

    安装 npm install --save react-wx-images-viewer 使用 import WxImageViewer from 'react-wx-images-viewer'; ...

  8. docker环境常用命令

    Ubuntu 安装docker及docker-compose 安装: apt-get install docker apt-get install docker-compose 启动docker环境: ...

  9. [PHP][linux] 命令行脚本接收传入参数的

    第一种 :用{ $argv }接受参数 第二种 : getopt() 第三种:

  10. curl发送多维数组

    //通过curl模拟post的请求: function SendDataByCurl($url,$data=array()){ //对空格进行转义 $url = str_replace(' ','+' ...