2棋盘覆盖(分治)

一、什么是棋盘覆盖?

在一个 2^k * 2^k 个方格组成的棋盘中,若恰有一个方格与其他方格不同,则称该方格为一个特殊方格,且称该棋盘为一个特殊棋盘。显然,特殊方格在棋盘上出现的位置有 4^k 种情况,即k>=0,有4^k种不同的特殊棋盘。

棋盘覆盖:用4种不同形态的L型骨牌覆盖一个给定的特殊棋盘(即特殊方格的位置已经确定了)上除特殊方格外的所有方格,且任何两个L型骨牌不得重复覆盖。按照规则,我们很容易知道,在2k*2k的棋盘覆盖中,用到的L型骨盘数恰为(4^k-1)/3,即(所有方格个数-特殊方格个数)/3
在这里插入图片描述

二、实现棋盘覆盖的思路和方法

棋盘覆盖实现的基本方法为分治法,是设计棋盘覆盖的一个简捷的算法,那什么是分治法?
分治法的基本思路:将一个规模为n的问题分解为k个规模较小的子问题,这些子问题相互独立且与原问题相同。递归地解决这些子问题,然后将各个子问题的解合并得到原问题的解。简单地说,就是将规模为n的问题自顶向下分解,直到子问题分解到足够小,可以容易解决时,再自底向上合并,从而得到原来的解。
分析思路:当k>0时,将2k*2k棋盘分割为4个2(k-1)*2(k-1)子棋盘,如图所示:
在这里插入图片描述
特殊方格必定位于这四个小棋盘中,其余三个子棋盘没有特殊方格,为了将这三个无特殊方格的子棋盘转换为特殊棋盘,我们可以用一个L型骨盘覆盖这三个较小棋盘的会合处,如图所示:
在这里插入图片描述 从图上可以看出,这三个子棋盘上被L型骨牌覆盖的方格就成为该棋盘上的特殊方格,从而将问题分解为4个较小规模的棋盘覆盖问题。递归地使用这种分割方法,直至棋盘简化为1*1棋盘,就结束递归。

实现这种算法的分析:每次都对分割后的四个小方块进行判断,判断特殊方格是否在里面。
这里的判断的方法是每次先记录下整个大方块的左上角方格的行列坐标,然后再与特殊方格坐标进行比较,就可以知道特殊方格是否在该块中。如果特殊方块在里面,这直接递归下去求即可,如果不在,这根据分割的四个方块的不同位置,把右下角、左下角、右上角或者左上角的方格标记为特殊方块,然后继续递归。在递归函数里,还要有一个变量s来记录边的方格数,每次对方块进行划分时,边的方格数都会减半,这个变量是为了方便判断特殊方格的位置。

三、棋盘覆盖的具体实现代码

#include<iostream>
using namespace std;
// tr 棋盘左上角方格的行号;  行row
// tc 棋盘左上角方格的列号;   列column 
// dr 特殊方格所在的行号;
// dc 特殊方格所在的列号;
// size =2^k,棋盘规模为2^k*2^k 
int Board[100][100];
int tile = 0; 
void ChessBoard(int tr, int tc, int dr, int dc, int size){
	if(size==1) return;
	int t = tile++; //L型骨牌号
	int s = size/2;
	if(dr<tr+s && dc<tc+s)          //特殊方格在此棋盘中  左上角方格 1 
		ChessBoard(tr,tc,dr,dc,s);
	else{                              //此棋盘中无特殊方格 
		Board[tr+s-1][tc+s-1] = t;     //用t号L型骨牌覆盖右下角 
		ChessBoard(tr,tc,tr+s-1,tc+s-1,s) ;//覆盖其余方格 
	} 
	if(dr<tr+s && dc>=tc+s)         //特殊棋盘在右上角方格      2
		ChessBoard(tr,tc+s,dr,dc,s); 
	else{
		Board[tr+s-1][tc+s]=t;      //用t号L型骨牌覆盖左下角 
		ChessBoard(tr,tc+s,tr+s-1,tc+s,s);
	} 
	if(dr>=tr+s && dc<tc+s)         //特殊棋盘在左下角方格      3
		ChessBoard(tr+s,tc,dr,dc,s);
	else{
		Board[tr+s][tc+s-1] = t;    //用t号L型骨牌覆盖右上角 
		ChessBoard(tr+s,tc,tr+s,tc+s-1,s); 
	} 
	if(dr>=tr+s && dc>=tc+s)        //特殊棋盘在右下角方格     4
		ChessBoard(tr+s,tc+s,dr,dc,s);
	else{
		Board[tr+s][tc+s] = t;      //用t号L型骨牌覆盖左上角 
		ChessBoard(tr+s,tc+s,tr+s,tc+s,s);
	}
} 
void Print(int size){
	for(int i=1;i<=size;i++){
		for(int j=1;j<=size;j++)
			cout<<Board[i][j]<<" "; 
		cout<<endl;
	}
}
int main(){
	int size;
	int dr,dc;
	cout<<"请输入棋盘的行列号:";
	cin>>size;
	cout<<"请输入特殊方格的行和列号";
	cin>>dr>>dc;
	cout<<endl;
	ChessBoard(1,1,dr,dc,size);
	Print(size);
	return 0;
}

在这里插入图片描述

  • 4
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

清木!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值