2017 蓝桥杯决赛 C++B(2)瓷砖样式 dfs + hash去重

2024-04-05 23:48

本文主要是介绍2017 蓝桥杯决赛 C++B(2)瓷砖样式 dfs + hash去重,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

标题:磁砖样式

小明家的一面装饰墙原来是 3*10 的小方格。
现在手头有一批刚好能盖住2个小方格的长方形瓷砖。
瓷砖只有两种颜色:黄色和橙色。
小明想知道,对于这么简陋的原料,可以贴出多少种不同的花样来。
小明有个小小的强迫症:忍受不了任何2*2的小格子是同一种颜色。
(瓷砖不能切割,不能重叠,也不能只铺一部分。另外,只考虑组合图案,请忽略瓷砖的拼缝)
显然,对于 2*3 个小格子来说,口算都可以知道:一共10种贴法,如【p1.png所示】
但对于 3*10 的格子呢?肯定是个不小的数目,请你利用计算机的威力算出该数字。

注意:你需要提交的是一个整数,不要填写任何多余的内容(比如:说明性文字)


答案: 101466


#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<map>
using namespace std;int mp[4][11];
map<int,int> hash;
int tol;bool noSame()//判断是否存在4个方块同样 {for(int i = 1; i <= 2; ++i){for(int j = 1; j <= 9; ++j){if(mp[i][j] == mp[i+1][j] && mp[i+1][j] == mp[i][j+1] && mp[i][j+1] == mp[i+1][j+1])return false;}}return true;}void color(int x, int y, int type,int v)//根据type判断水平2块还是数值2块  染成 v {if(type == 1){ //水平 for(int i = x; i < x+1; ++i){for(int j = y; j < y+2; ++j){mp[i][j] = v;}}}else if(type == 2){for(int i = x; i < x+2; ++i){for(int j = y; j < y+1; ++j){mp[i][j] = v;}}}}
bool checkX(int x, int y)//判断X方向是否能放2块 {if(x+1 > 3 || y > 10)return false;for(int i = x; i < x+2; ++i){for(int j = y; j < y+1; ++j){if(mp[i][j])return false;}}return true;}void output(){for(int i = 1; i <= 3; ++i){for(int j = 1; j <= 10; ++j){printf("%d",mp[i][j]);}printf("\n");}}	bool checkY(int x, int y)//判断y方向能否放2块 {if(x > 3 || y+1 > 10)return false;for(int i = x; i < x+1; ++i){for(int j = y; j < y+2; ++j){if(mp[i][j])return false;}}	return true;}void dfs(int x, int y){if(x > 3){if(noSame()){int bit = 1;long long ans = 0;for(int i = 1; i <= 3; ++i){for(int j = 1; j <= 10; ++j){ans += bit*mp[i][j];//用ans来标识唯一的涂色方案bit <<= 1;}}if(!hash[ans]){++tol;++hash[ans];}}return ;}	if(mp[x][y] == 0){if(checkX(x,y)){for(int i = 1; i <= 2; ++i){//注意瓷砖有2种 可以染不同颜色 color(x,y,2,i);if(y == 10){dfs(x+1,1);}else{dfs(x,y+1);}color(x,y,2,0);}}if(checkY(x,y)){for(int i = 1; i <= 2; ++i){color(x,y,1,i);if(y == 10){dfs(x+1,1);}else{dfs(x,y+1);}color(x,y,1,0);}}}else{if(y == 10){dfs(x+1,1);}else{dfs(x,y+1);}}}int main(){memset(mp,0,sizeof(mp));dfs(1,1);printf("%d",tol);return 0;}

这篇关于2017 蓝桥杯决赛 C++B(2)瓷砖样式 dfs + hash去重的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/878486

相关文章

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

hdu1496(用hash思想统计数目)

作为一个刚学hash的孩子,感觉这道题目很不错,灵活的运用的数组的下标。 解题步骤:如果用常规方法解,那么时间复杂度为O(n^4),肯定会超时,然后参考了网上的解题方法,将等式分成两个部分,a*x1^2+b*x2^2和c*x3^2+d*x4^2, 各自作为数组的下标,如果两部分相加为0,则满足等式; 代码如下: #include<iostream>#include<algorithm

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

usaco 1.2 Milking Cows(类hash表)

第一种思路被卡了时间 到第二种思路的时候就觉得第一种思路太坑爹了 代码又长又臭还超时!! 第一种思路:我不知道为什么最后一组数据会被卡 超时超了0.2s左右 大概想法是 快排加一个遍历 先将开始时间按升序排好 然后开始遍历比较 1 若 下一个开始beg[i] 小于 tem_end 则说明本组数据与上组数据是在连续的一个区间 取max( ed[i],tem_end ) 2 反之 这个