NC19910 [CQOI2007]矩形RECT
题目
题目描述
给一个a*b矩形,由a*b个单位正方形组成。你需要沿着网格线把它分成分空的两部分,每部分所有格子连通,且至少有一个格子在原矩形的边界上。“连通”是指任两个格子都可以通过水平或者竖直路径连在一起。 求方案总数。例如3*2的矩形有15种方案。
输入描述
输入仅一行,为两个整数a,b。 ,
输出描述
输出仅一行,即方案总数。
示例1
输入
3 2
输出
15
示例2
输入
3 3
输出
52
题解
知识点:DFS。
计数问题用dfs较为合适,注意到只要切成两块,因此切入点和切出点各仅有一个,而且切痕不能交叉。因此枚举各边的切入点,搜索所有切痕条数,切出边一次算一条(包括自己边)。
由于枚举时会产生重复情况,因为路径的终点也能作为起点返回去算一条,但结合矩形的对称性,我们枚举横竖两边即可。先给边标号 ,假设 是横竖两边,那么能搜索出 边上所有点的切线条数,其中 有两次是因为自己边作为起点和终点可以有往返两条路径。我们把其中 作为 , 作为 , 作为 即可有边的全部组合。
时间复杂度
空间复杂度
代码
#include <bits/stdc++.h> using namespace std; int n, m; bool vis[7][8]; const int dir[4][2] = { {1,0},{-1,0},{0,1},{0,-1} }; int cnt; void dfs(int x, int y) { if (!x || !y || x == n || y == m) { cnt++; return; } for (int i = 0;i < 4;i++) { int xx = x + dir[i][0]; int yy = y + dir[i][1]; if (vis[xx][yy]) continue; vis[xx][yy] = 1; dfs(xx, yy); vis[xx][yy] = 0; } } int main() { std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0); cin >> n >> m; ///只需枚举横竖两条边,因为自己一边或者横竖两边之间任意路径,都会有一个重复的返回路径 ///根据对称性,可以看作对边自己或者对边横竖之间的所有路径 for (int i = 1;i < n;i++) { vis[i][0] = 1; vis[i][1] = 1; dfs(i, 1); vis[i][0] = 0; vis[i][1] = 0; } for (int i = 1;i < m;i++) { vis[0][i] = 1; vis[1][i] = 1; dfs(1, i); vis[0][i] = 0; vis[1][i] = 0; } cout << cnt << '\n'; return 0; }
本文来自博客园,作者:空白菌,转载请注明原文链接:https://www.cnblogs.com/BlankYang/p/16482908.html
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧