分而治之法解决问题.docVIP

  1. 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
  2. 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  3. 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  4. 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  5. 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  6. 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  7. 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
201100820157 数学与统计学院 全秋洪 残缺棋盘问题 残缺棋盘(defective chess board)是一个有2k×2k个方格的棋盘,其中恰有一个方格残缺。图3给出k≤2时各种可能的残缺棋盘,其中残缺的方格用阴影表示。注意当k=0时,仅存在一种可能的残缺棋盘(如图3a所示)。事实上,对于任意k,恰好存在22k种不同的残缺棋盘。 残缺棋盘的问题要求用3个方格的板(三格板)(triominoes)覆盖残缺棋盘(如图4所示)。在此覆盖中,两个三格板不能重叠,三格板不能覆盖残缺方格,但必须覆盖其他所有的方格。在这种限制条件下,所需要的三格板总数为(22k-1)/3。可以验证(22k-1)/3是一个整数。k为0的残缺棋盘很容易被覆盖,因为它没有非残缺的方格,用于覆盖的三格板的数目为0。当k=1时,正好存在3个非残缺的方格,并且这三个方格可用图4中的某一方向的三格板来覆盖。 实际是要求用三格板覆盖n*n的方格棋盘,空出指定位置。 图3 残缺棋盘 图4 不同方向的三格板 用分而治之方法可以很好地解决残缺棋盘问题。这一方法可将覆盖2k×2k残缺棋盘的问题转化为覆盖较小残缺棋盘的问题。2k×2k棋盘一个很自然的划分方法就是将它划分为如图5a所示的4个2k-1×2 k-1棋盘。注意到当完成这种划分后,4个小棋盘中仅仅有一个棋盘存在残缺方格(因为原来的2k×2k棋盘仅仅有一个残缺方格)。首先覆盖其中包含残缺方格的2k-1×2 k-1残缺棋盘,然后把剩下的3个小棋盘转变为残缺棋盘,为此将一个三格板放在由这3个小棋盘形成的角上,如图5b所示,其中原2k×2k棋盘中的残缺方格落入左上角的2k-1×2 k-1棋盘。可以采用这种分割技术递归地覆盖2k×2k残缺棋盘。当棋盘的大小减为1×1时,递归过程终止。此时1×1的棋盘中仅仅包含一个方格且此方格残缺,所以无需放置三格板。 图5 分割2k×2k棋盘 可以将上述分而治之算法编写成一个递归的C++函数TileBoard。该函数定义了一个全局的二维整数数组变量Board来表示棋盘。Board[0][0]表示棋盘中左上角的方格。该函数还定义了一个全局整数变量tile,其初始值为1。函数的输入参数如下: ?tr棋盘中左上角方格所在行。 ?tc棋盘中左上角方格所在列。 ?dr残缺方块所在行。 ?dc残缺方块所在列。 ?size棋盘的行数或列数。 TileBoard函数的调用格式为TileBoard(tr,tc,dr,dc,size),其中size=2k。覆盖残缺棋盘所需要的三格板数目为(size2-1)/3。函数TileBoard用整数1到(size2-1)/3来表示这些三格板,并用三格板的标号来标记被该三格板覆盖的非残缺方格。 令t(k)为函数TileBoard覆盖一个2k×2k残缺棋盘所需要的时间。当k=0时,size等于1,覆盖它将花费常数时间d。当k0时,将进行4次递归的函数调用,这些调用需花费的时间为4t(k-1)。除了这些时间外,if条件测试和覆盖3个非残缺方格也需要时间,假设用常数c表示这些额外时间。可以得到以下递归表达式: 可以用迭代的方法来计算这个表达式,可得t(k)=(22k)。 程序 :残缺棋盘 #include stdio.h void TileBoard(int tr,int tc,int dr,int dc,int size); void OutputBoard(int size); const n=8; int tile=1; int Board[n][n]; void main() { TileBoard( 0, 0, 3, 2, n ); OutputBoard( n ); } void TileBoard(int tr,int tc,int dr,int dc,int size) { //覆盖残缺棋盘 if(size==1) return; int t=tile++, //所使用的三格板的数目 s=size/2;//象限大小 //覆盖左上象限 if(drtr+sdctc+s) //残缺方格位于本象限 TileBoard(tr,tc,dr,dc,s); else { //本象限中没有残缺方格, 把三格板t放在右下角 Board[tr+s-1][tc+s-1]=t; //覆盖其余部分 TileBoard(tr,tc,tr+s-1,tc+s-1,s); } //覆盖右上象限 if(drtr+sdc=tc+s) //残缺方格位于本象限 TileBoard(tr,tc+s,dr,dc,s); else { //本象限中没有残缺方格, 把三格板t放在左下角 Board[tr+s-1][tc+

文档评论(0)

mmc566887 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档