原创


标题:磁砖样式

小明家的一面装饰墙原来是 3*10 的小方格。
现在手头有一批刚好能盖住2个小方格的长方形瓷砖。
瓷砖只有两种颜色:黄色和橙色。

小明想知道,对于这么简陋的原料,可以贴出多少种不同的花样来。
小明有个小小的强迫症:忍受不了任何2*2的小格子是同一种颜色。
(瓷砖不能切割,不能重叠,也不能只铺一部分。另外,只考虑组合图案,请忽略瓷砖的拼缝)
显然,对于 2*3 个小格子来说,口算都可以知道:一共10种贴法,如【p1.png所示】

但对于 3*10 的格子呢?肯定是个不小的数目,请你利用计算机的威力算出该数字。

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

一开始的想法是在数组里面枚举出全部的情况组合,比如2*3的方格,用0/1代表两种颜色;

那么一共会有pow(2,6)=64种情况,然后用3个判断条件筛选出符合要求的贴砖方式:

1:   数码0/1有偶数个;

2:任意2*2的方格的数码不能相同;

3:任意一个方格在其相邻的(上/下/左/右)方格至少有一个相同的数码;

代码如下:

 #include<stdio.h>

 int arr[][]={};
int count=; int Judge(){
int i=;
int j=;
//条件1:偶数个0和1
int count_zero=; //存储0/1个数
int count_one=;
for(i=;i<=;i++){
for(j=;j<=;j++){
if(arr[i][j]==){
count_zero++;
}
else{
count_one++;
}
}
}
if(count_zero%!= || count_one%!=){
return ;
}
//条件2:2*2方格的数字都不相同
int x=;
int y=;
for(x=;x<=;x++){ //循环至行数-1
for(y=;y<=;y++){ //循环至列数-1
int a=arr[x][y];
int b=arr[x][y+];
int c=arr[x+][y];
int d=arr[x+][y+];
if(a==b && a==c && a==d && b==c && b==d && c==d)
return ;
}
}
//条件3:每个数的相邻位置要有与其相同的数
for(i=;i<=;i++){
for(j=;j<=;j++){
int value=arr[i][j];
if(i->=){ //上
if(value==arr[i-][j]){
continue;
}
}
if(i+<=){ //下
if(value==arr[i+][j]){
continue;
}
}
if(j->=){ //左
if(value==arr[i][j-]){
continue;
}
}
if(j+<=){ //右
if(value==arr[i][j+]){
continue;
}
else{ //没有相邻的数码
return ;
}
}
}
}
return ;
} void Style(int i,int j){ //i行、j列 if(i== && j==){ //得到一种贴砖方式 if(Judge()==){
/*
int a=0; //输出
int b=0;
for(a=0;a<=1;a++){
for(b=0;b<=2;b++){
printf("%d ",arr[a][b]);
if(b==2){
printf("\n");
}
}
}
*/
count++;
}
return;
} if(j==){
i++;
j=;
}
int v=;
for(v=;v<=;v++){ //每个位置0-1循环
arr[i][j]=v;
Style(i,j+);
arr[i][j]=; //回溯
}
} int main(){
Style(,);
printf("%d",count);
return ;
}

只检验了2*3、3*6的方格的样例,2*3的样例输出了正确的答案,3*6的输出错误。

3*10的数据量太大跑不出来了。

上面的3个控制条件太少,像

1 0 1 1 0 0

1 1 0 0 0 1

1 0 1 1 0 1

这样的贴砖方式能通过条件但却是不符合要求的,因为这种方式太过于暴力,所以没有继续改进。

此题应该通过DFS解决:

3*10的方格,每个空方格都可以有4种贴法:(我们以1/2号定义两种颜色的砖)

横着贴1号砖、横着贴2号砖、竖着贴1号砖、竖着贴2号砖

所以我们用DFS搜索每块空砖的这4种贴法即可。

 #include<stdio.h>
#define row 3
#define rank 10 int count=;
int arr[row+][rank+]={}; //--------------① int Judge(int x,int y){ //每一块砖的左上、右上、左下、右下四个2*2方格
if(arr[x][y]==arr[x-][y] && arr[x][y]==arr[x-][y-] && arr[x][y]==arr[x][y-]){ //左上
return ;
}
if(arr[x][y]==arr[x-][y] && arr[x][y]==arr[x-][y+] && arr[x][y]==arr[x][y+]){ //右上
return ;
}
if(arr[x][y]==arr[x][y-] && arr[x][y]==arr[x+][y-] && arr[x][y]==arr[x+][y]){ //左下
return ;
}
if(arr[x][y]==arr[x][y+] && arr[x][y]==arr[x+][y] && arr[x][y]==arr[x+][y+]){ //右下
return ;
}
return ;
} void dfs(int x,int y){
if(x== && y==){
count++;
return;
}
if(y==){
dfs(x+,);
return;
}
if(arr[x][y]==-){ //4种铺法可以任意顺序
if(arr[x][y+]==-){ // 横铺1
arr[x][y]=;
arr[x][y+]=;
if(Judge(x,y)==){
dfs(x,y+);
}
arr[x][y]=-;
arr[x][y+]=-;
}
if(arr[x+][y]==-){ // 竖铺2
arr[x][y]=;
arr[x+][y]=;
if(Judge(x,y)==){
dfs(x,y+);
}
arr[x][y]=-;
arr[x+][y]=-;
}
if(arr[x+][y]==-){ // 竖铺1
arr[x][y]=;
arr[x+][y]=;
if(Judge(x,y)==){
dfs(x,y+);
}
arr[x][y]=-;
arr[x+][y]=-;
}
if(arr[x][y+]==-){ // 横铺2
arr[x][y]=;
arr[x][y+]=;
if(Judge(x,y)==){
dfs(x,y+);
}
arr[x][y]=-;
arr[x][y+]=-;
}
} else{
dfs(x,y+);
}
} int main(){
int i=;
int j=;
for(i=;i<=;i++){ //-------------②
for(j=;j<=;j++){
arr[i][j]=-;
}
}
dfs(,);
printf("%d",count);
return ;
}

对代码中①/②的解释:

①:申请5*12的空间为方便对3*10的方格进行2*2的判断

②:只能对3*10的方格进行赋值,保证第0/4行、第0/11列(即外围一圈)的值和里面3*10的方格不同,具有很大的便利性(请大家慢慢体会)。

答案:114434

3:38:00

2018-05-07

磁砖样式——第八届蓝桥杯C语言B组(国赛)第二题的更多相关文章

  1. 2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告

    2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告 勘误1:第6题第4个 if最后一个条件粗心写错了,答案应为1580. 条件应为abs(a[3]-a[7])!=1,宝宝心理苦啊.!感谢zzh ...

  2. 发现环——第八届蓝桥杯C语言B组(国赛)第四题

    原创 标题:发现环 小明的实验室有N台电脑,编号1~N.原本这N台电脑之间有N-1条数据链接相连,恰好构成一个树形网络.在树形网络上,任意两台 电脑之间有唯一的路径相连. 不过在最近一次维护网络时,管 ...

  3. 希尔伯特曲线——第八届蓝桥杯C语言B组(国赛)第三题

    原创 标题:希尔伯特曲线 希尔伯特曲线是以下一系列分形曲线 Hn 的极限.我们可以把 Hn 看作一条覆盖 2^n × 2^n 方格矩阵的曲线,曲线上一共有 2^n × 2^n 个顶点(包括左下角起点和 ...

  4. 2017第八届蓝桥杯C/C++ B组省赛-日期问题

    标题:日期问题 小明正在整理一批历史文献.这些历史文献中出现了很多日期.小明知道这些日期都在1960年1月1日至2059年12月31日.令小明头疼的是,这些日期采用的格式非常不统一,有采用年/月/日的 ...

  5. 2017年第八届蓝桥杯C/C++B组省赛题目解析

    一. 购物单 小明刚刚找到工作,老板人很好,只是老板夫人很爱购物.老板忙的时候经常让小明帮忙到商场代为购物.小明很厌烦,但又不好推辞. 这不,XX大促销又来了!老板夫人开出了长长的购物单,都是有打折优 ...

  6. 2017第八届蓝桥杯C/C++ B组省赛-购物单

    标题: 购物单 小明刚刚找到工作,老板人很好,只是老板夫人很爱购物.老板忙的时候经常让小明帮忙到商场代为购物.小明很厌烦,但又不好推辞. 这不,XX大促销又来了!老板夫人开出了长长的购物单,都是有打折 ...

  7. 2017第八届蓝桥杯C/C++ B组省赛-等差素数列

    标题:等差素数列 2,3,5,7,11,13,....是素数序列. 类似:7,37,67,97,127,157 这样完全由素数组成的等差数列,叫等差素数数列. 上边的数列公差为30,长度为6. 200 ...

  8. 第八届蓝桥杯C/C++ B组省赛----分巧克力

    分巧克力 问题描述 儿童节那天有K位小朋友到小明家做客.小明拿出了珍藏的巧克力招待小朋友们. 小明一共有N块巧克力,其中第i块是Hi x Wi的方格组成的长方形. 为了公平起见,小明需要从这 N 块巧 ...

  9. 【蓝桥杯】2018年第九届蓝桥杯C/C++B组省赛——B题 等差素数列

    题目 标题:等差素数列 2,3,5,7,11,13,....是素数序列. 类似:7,37,67,97,127,157 这样完全由素数组成的等差数列,叫等差素数数列. 上边的数列公差为30,长度为6. ...

随机推荐

  1. [Swift] 创建一个对象

    创建一个对象 先写一个People类 // // People.swift // Class // // Created by YouXianMing on 15/3/18. // Copyright ...

  2. spider-抓取网页内容

    使用urllib2抓取网页内容: import urllib2 from HTMLParser import HTMLParser request = urllib2.Request('http:// ...

  3. Python学习---django模板语法180122

    django模板语法[Template] 模版的组成:  HTML代码+逻辑控制代码  <h1> {{ user_name }} </h1> 逻辑控制代码的组成: 1.变量: ...

  4. 3D旋转相册的实现

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  5. 原生ajax和jsonp

    封装方法: function ajax(options) { options = options || {}; options.type = (options.type || "GET&qu ...

  6. December 06th 2016 Week 50th Tuesday

    Behind every beautiful thing, there is some kind of pain. 美丽背后,必有努力. No pains, no gains. But it seem ...

  7. HDU 1542 矩形面积并

    推荐阅读这篇文章 这里仅根据上述文章进行一些补充.主要是注意这里线段树点的意义变成了“边”.比如+-----+-----+,看作3个点abc和两条边e1和e2,那么线段树中点a代表e1,点b代表e2. ...

  8. codeforces 932E Team Work(组合数学、dp)

    codeforces 932E Team Work 题意 给定 \(n(1e9)\).\(k(5000)\).求 \(\Sigma_{x=1}^{n}C_n^xx^k\). 题解 解法一 官方题解 的 ...

  9. 如何批量下载bing的背景图片?

    工具准备 wget(点击下载) 批处理命令(点击下载) 网友提供的接口:http://area.sinaapp.com/bingImg?daysAgo=1(1代表天数) 实现步骤 1.打开记事本,并将 ...

  10. Kali-linux使用Aircrack-ng工具破解无线网络

    Aircrack-ng是一款基于破解无线802.11协议的WEP及WPA-PSK加密的工具.该工具主要用了两种攻击方式进行WEP破解.一种是FMS攻击,该攻击方式是以发现该WEP漏洞的研究人员名字(S ...