题意转化

考虑我们对于集合中每一个\(i\),若\(i-2,i+k\)存在,就向其连边。

那么,一个合法的集合就需要满足,不会存在环。

这样问题转化到了图上,就变得具体了许多,也就更容易考虑、求解了。

奇偶性讨论

这题对于\(k\)为奇数/偶数的情况,要分别处理。

由于偶数情况较为简单,所以我们从偶数讲起。

当\(k\)为偶数

这时我们发现奇数和偶数是独立的。

我们分别求出奇数和偶数的方案数(\(DP(\lfloor\frac{n+1}2\rfloor,\frac k2)\)和\(DP(\lfloor\frac n2\rfloor,\frac k2)\)),然后乘起来即为总方案数。

而此时,原先的\(i-2\)现在就变成了\(i-1\),因此只要不连续选择\(k+1\)个数即可。

那么就很简单了,设\(f_{i,j}\)表示当前第\(i\)个数,已连续选择了\(j\)个数,转移分选不选讨论。

当\(k\)为奇数

可以自己在草稿纸上画个图,画两列数,一列奇数,一列偶数,其中左边每个数\(i\)与右边\(i+k\)对齐,然后连上边。

然后再对着图研究下就可以发现,若能从右边某个数开始,只往下/左两个方向走,连续选择\(k+2\)个数,就不合法了。

那么我们直接设\(f_{i,j,k}\)表示当前第\(i\)行,最多连续选的数为\(j\)个,右边已连续选择\(k\)个数,转移分左右两边选不选共四种情况讨论。

提示一下,这里用刷表法\(DP\)似乎比较方便。

代码

#include<bits/stdc++.h>
#define Tp template<typename Ty>
#define Ts template<typename Ty,typename... Ar>
#define Reg register
#define RI Reg int
#define Con const
#define CI Con int&
#define I inline
#define W while
#define N 150
#define Inc(x,y) ((x+=(y))>=X&&(x-=X))
using namespace std;
int n,m,X;
class EvenSolver//k为偶数
{
private:
int f[N+5][N+5];
I int DP(CI n,CI m)//处理小问题
{
RI i,j,ans=0;memset(f,0,sizeof(f));//清空
for(f[0][0]=1,i=0;i^n;++i) for(j=0;j<=m;++j) Inc(f[i+1][0],f[i][j]),Inc(f[i+1][j+1],f[i][j]);//DP
for(i=0;i<=m;++i) Inc(ans,f[n][i]);return ans;//统计答案
}
public:
I void Solve() {printf("%d",1LL*DP(n+1>>1,m>>1)*DP(n>>1,m>>1)%X);}
}E;
class OddSolver//k为奇数
{
private:
int f[N+5][N+5][N+5];
public:
I void Solve()
{
RI i,j,k,ans=0;f[0][0][0]=1;//初始化
for(i=0;i<(m>>1)+(n+1>>1);++i) for(j=0;j<=(n>>1);++j) for(k=0;k<=m+1;++k)//DP
Inc(f[i+1][0][0],f[i][j][k]),i<(n>>1)&&Inc(f[i+1][j+1][0],f[i][j][k]),//两边都不选,左选右不选
i>=(m>>1)&&Inc(f[i+1][0][k?k+1:0],f[i][j][k]),//左不选右选
i>=(m>>1)&&i<(n>>1)&&Inc(f[i+1][j+1][max(j+2,k+1)],f[i][j][k]);//左右都选
for(j=0;j<=(n>>1);++j) for(k=0;k<=m+1;++k) Inc(ans,f[(m>>1)+(n+1>>1)][j][k]);//统计答案
printf("%d",ans);//输出答案
}
}O;
int main()
{
freopen("seventeen.in","r",stdin),freopen("seventeen.out","w",stdout);
return scanf("%d%d%d",&n,&m,&X),m&1?O.Solve():E.Solve(),0;
}

最新文章

  1. sizzle分析记录:属性选择器
  2. Spring的注解方式实现AOP
  3. ListView之头部浮动效果
  4. Dynamic CRM 2013学习笔记(二十八)用JS动态设置字段的change事件、必填、禁用以及可见
  5. JavaScript实例-----反选
  6. CLI-error
  7. COJ 0017 20604悲剧文本
  8. 無心插柳的Linux學習者代言人——蔡德明
  9. RDLC 聚合函数按条件求和,显示&quot;错误号&quot;
  10. Typecho中文验证码Captcha插件
  11. oracle学习笔记(三)oracle函数
  12. JS数组及内置对象
  13. 解决MySQL中文乱码问题
  14. Kafka设计解析(八)- Exactly Once语义与事务机制原理
  15. Beta阶段敏捷冲刺报告-DAY4
  16. Android Studio 2.2 新功能详解
  17. sql中1=1的and和or问题
  18. 树莓派3B+ HDMI连接显示屏 因供电问题而不能进入系统
  19. 纽约工作日志流水账 Day 2
  20. myeclipse 从数据库生成java实体类

热门文章

  1. CF1269A Equation
  2. go-面向对象编程(上)
  3. java基础(22):File、递归
  4. Java生鲜电商平台-商品分类表和商品类型表的区别与数据库设计
  5. 如何在CAD中批量打印图纸?这种方法你要知道
  6. Spring 注解配置Bean
  7. 后端返回null,前端怎么处理?数据容错——不用过分相信外部数据
  8. kali linux查看局域网下所有IP,并对指定IP实施局域网内攻击(断网,随时查看对方密码,上网痕迹等)
  9. nginx在centos下的安装
  10. Python正则、re模块