P10681 COTS/CETS 2024 奇偶矩阵 Tablica

来自 qnqfff 大佬的梦幻 dp。

约定

二元组 \((n,m)\) 表示一个 \(n\) 行 \(m\) 列的矩形。

不添加说明的子问题,限制与题面一致。

思路

先考虑放最后一行,发现你填的位置经过变换后可以得到其他的结果,也就是说只要乘上变换的方案数就可以任选位置放。

那么考虑填一个还是两个。

1.填一个

填在左下角,不考虑填的那一列(填的列不再填)变成了 \((n-1,m-1)\) 的子问题;考虑那一列,变成了一个 \((m,n-1)\) 其中有一行填一个的子问题。

2.填两个

填在左下角两个:

  • 2.1 不考虑填的列,是一个 \((n-1,m-2)\) 的子问题。
  • 2.2 考虑填的两列,是一个 \((m,n-1)\) 其中两行有一行(不填状况已经考虑,这里必填一个可以更多)必填一个的子问题。

3.一行填一个

填左下角,是一个 \((n-1,m-1)\) 的子问题和 \((m,n-1)\) 其中一行填一个的子问题(该行填完后对于其他列来说又只能填一个)。

4.两行必填一个(有一行可填可不填)

  • 4.1 同一列填一个 \((n-2,m-1)\) 的子问题和 \((m,n-2)\) 其中一行填一个的子问题。
  • 4.2 同一列填两个 \((n-2,m-1)\) 的子问题。
  • 4.3 任选两列列填两个 \((n-2,m-2)\) 的子问题和 \((m,n-2)\) 其中两行必填一个的子问题。

在大力分讨之后,发现所有情况都可以分为子问题,而子问题合并的系数也易于得出,具体系数参考代码或读者自行思考。

使用记忆化搜索实现,较为容易。

#include<bits/stdc++.h>
using namespace std; #define ll long long const ll mod=1e9+7;
// const ll mod=1211221111;
const int maxn=3005; int n,m; ll dp[maxn][maxn],f[maxn][maxn],g[maxn][maxn]; inline ll dfs(int n,int m);
int go(int n,int m){//求 (n,m) 中某一行必填一个的方案数
if(n<=0||m<=0) return 0;
if(~f[n][m]) return f[n][m];
if(m==1)
{
if(n<=2) return f[n][m]=1;
return f[n][m]=0;
}
return f[n][m]=(dfs(n-1,m-1)+go(m,n-1))*m%mod;
}
inline ll _go(int n,int m)//求 (n,m) 中某两行有一行必填一个的方案数
{
if(n<=0||m<=0) return 0;
if(~g[n][m]) return g[n][m];
if(m==1)
{
if(n==2) return g[n][m]=3;
else if(n==3) return g[n][m]=2;
return g[n][m]=0;
}
if(m==2)
{
g[n][m]=((dfs(n-2,m-1)+go(m,n-2))*2%mod*m%mod/*同一列填一个*/+dfs(n-2,m-1)*m%mod)%mod;/*同一列填一个*/
if(n<=4)//直接算两列的贡献
{
if(n==2) g[n][m]+=2;
else if(n==3) g[n][m]+=6;
else g[n][m]+=4;
}
return g[n][m]=0;
}
return g[n][m]=(((dfs(n-2,m-1)+go(m,n-2))*2%mod*m%mod/*同一列填一个*/+dfs(n-2,m-1)*m%mod/*同一列填两个*/)+(_go(m,n-2)+dfs(n-2,m-2))*m%mod*(m-1)%mod/*任选两列列填两个(有序)*/)%mod;
}
inline ll dfs(int n,int m)//求子问题(n,m)的方案数
{
if(n<=0||m<=0) return 0;
if(~dp[n][m]) return dp[n][m];
if(2*n<m||2*m<n) return dp[n][m]=0;
if(n==1||m==1) return dp[n][m]=1;
return dp[n][m]=((dfs(n-1,m-1)+go(m,n-1))*m%mod/*填一个*/+(dfs(n-1,m-2)+_go(m,n-1))*(m*(m-1)/2)%mod/*填两个*/)%mod;
}
inline void solve()
{
memset(dp,-1,sizeof(dp));memset(f,-1,sizeof(f));memset(g,-1,sizeof(g));
printf("%lld",dfs(n,m));
} int main()
{
scanf("%d%d",&n,&m);
if(2*n<m) printf("0"),exit(0);
if(n==1) printf("1"),exit(0);
solve();
}

P10681 COTS/CETS 2024 奇偶矩阵 Tablica的更多相关文章

  1. HDOJ-1010 Tempter of the Bone(dfs+剪枝)

    http://acm.hdu.edu.cn/showproblem.php?pid=1010 给出一个n*m的迷宫 X为墙 .为空地 S为起点 D为终点 给出时间T 每走一步花费一单位的时间 走过的空 ...

  2. 网络流专题练习Day2

    04/17  目前做了:题 由于目前六道都是1A感觉非常爽... BZOJ1412: [ZJOI2009]狼和羊的故事 “狼爱上羊啊爱的疯狂,谁让他们真爱了一场:狼爱上羊啊并不荒唐,他们说有爱就有方向 ...

  3. HDU 1010 Tempter of the Bone(DFS+奇偶剪枝)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1010 题目大意: 输入 n m t,生成 n*m 矩阵,矩阵元素由 ‘.’ 'S' 'D' 'X' 四 ...

  4. hdu.1043.Eight (打表 || 双广 + 奇偶逆序)

    Eight Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Sub ...

  5. Dynamic Morphing Square(动态变形矩阵)

    题目描述: 解题思路: 先对输入的N进行判断,是否不小于3,如果小于3,需要继续输入一个新的数,知道输入的N比3大. 第一个打印的矩阵,*号为最外面一圈,其余全为-. 第二个打印的矩阵,*号向内缩减了 ...

  6. hdu 1010:Tempter of the Bone(DFS + 奇偶剪枝)

    Tempter of the Bone Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  7. [Lonlife1031]Bob and Alice are eating food(递推,矩阵快速幂)

    题目链接:http://www.ifrog.cc/acm/problem/1031 题意:6个水果中挑出n个,使得其中2个水果个数必须是偶数,问有多少种选择方法. 设中0代表偶数,1代表奇数.分别代表 ...

  8. bzoj 1002 [FJOI2007]轮状病毒 高精度&&找规律&&基尔霍夫矩阵

    1002: [FJOI2007]轮状病毒 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 2234  Solved: 1227[Submit][Statu ...

  9. hdoj 1010 Tempter of the Bone【dfs查找能否在规定步数时从起点到达终点】【奇偶剪枝】

    Tempter of the Bone Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  10. [POJ 3734] Blocks (矩阵高速幂、组合数学)

    Blocks Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 3997   Accepted: 1775 Descriptio ...

随机推荐

  1. LeetCode300.最长递增子序列

    LeetCode300.最长递增子序列 力扣题目链接(opens new window) 给你一个整数数组 nums ,找到其中最长严格递增子序列的长度. 子序列是由数组派生而来的序列,删除(或不删除 ...

  2. Drools决策表实践运用

    Drools 决策表的使用与说明 Drools决策表的使用 官方文档决策表说明 决策表使用方式 执行drl代码及结果 Drools决策表的使用 官方文档决策表说明 Drools 决策表的使用 16.7 ...

  3. pikachu靶场 越权(水平越权+垂直越权)

    水平越权 A用户和B用户属于同一级别用户,但各自不能操作对方个人信息.A用户如果越权操作B用户个人信息的情况称为水行越权操作 三个用户 lucy/lili/kobe  密码都为123456 随便登录其 ...

  4. 扩展KMP (ex_KMP)

    一些约定: 字符串下标从1开始 s[1,i]表示S的第一个到第i个字符组成的字符串 解决的题型: 给你两个字符串A,B(A.size()=n,B.size()=m),求p数组 p[i]表示最大的len ...

  5. 深入理解Argo CD工作原理

    1. ArgoCD 的架构 ArgoCD 是一个 Kubernetes 原生的持续交付工具,它通过监控 Git 仓库中的应用定义来自动部署应用到 Kubernetes 集群.其核心架构由以下几个关键组 ...

  6. MyBatis——案例——删除(单个删除与批量删除)

    删除一个   1.编写接口方法:Mapper接口     参数:id     结果:void /** * 删除 */ int deleteById(int id);   2.编写sql语句:SQL映射 ...

  7. Hugging Face 论文平台 Daily Papers 功能全解析

    文/ Adeena, 在快速发展的研究领域,保持对最新进展的关注至关重要.为了帮助开发者和研究人员跟踪 AI 领域的前沿动态,Hugging Face 推出了 Daily Papers 页面.自发布以 ...

  8. HEOI2024 题目转存

    赛时测试数据下载 wind xor wormhole maze timeline sleep 题解参考 [省选联考 2024] 季风 题目背景 生活在二维平面的小 X 准备拜访小 Y,但由于气候的变化 ...

  9. php7新内容总结(随时更新)

    一.参数和返回值类型申明 可以申明的有:float,int,bool,string,interfaces,array,callable 一般模式: function sum(int ...$ints) ...

  10. ARM SMMU 与 IOMMU 的区别

    ARM SMMU (System Memory Management Unit) 和 IOMMU (Input-Output Memory Management Unit) 都是用于管理系统内存访问和 ...