QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 3749|回复: 3
打印 上一主题 下一主题

残缺棋盘

[复制链接]
字体大小: 正常 放大
韩冰        

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

跳转到指定楼层
1#
发表于 2004-10-4 05:16 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
<b>残缺棋盘</b>
5 O2 z- b4 t0 q6 r$ |: K) y8 j<>残缺棋盘(defective chessboard)是一个有2k×2k 个方格的棋盘,其中恰有一个方格残缺。图2 - 3给出k≤2时各种可能的残缺棋盘,其中残缺的方格用阴影表示。注意当k= 0时,仅存在一种可能的残缺棋盘(如图1 4 - 3 a所示)。事实上,对于任意k,恰好存在22k 种不同的残缺棋盘。
7 x5 c! l( g, i4 b7 S3 s; k! p, `8 P* g5 [* j. S
残缺棋盘的问题要求用三格板(t r i o m i n o e s)覆盖残缺棋盘(如图1 4 - 4所示)。在此覆盖中,两个三格板不能重叠,三格板不能覆盖残缺方格,但必须覆盖其他所有的方格。在这种限制条件下,所需要的三格板总数为( 22k -1 ) / 3。可以验证( 22k -1 ) / 3是一个整数。k 为0的残缺棋盘很容易被覆盖,因为它没有非残缺的方格,用于覆盖的三格板的数目为0。当k= 1时,正好存在3个非残缺的方格,并且这三个方格可用图1 4 - 4中的某一方向的三格板来覆盖。
0 \) q, U5 f, K
! V9 b  d; ^  Z' t用分而治之方法可以很好地解决残缺棋盘问题。这一方法可将覆盖2k×2k 残缺棋盘的问题转化为覆盖较小残缺棋盘的问题。2k×2k 棋盘一个很自然的划分方法就是将它划分为如图1 4 - 5 a所示的4个2k - 1×2k - 1 棋盘。注意到当完成这种划分后, 4个小棋盘中仅仅有一个棋盘存在残缺方格(因为原来的2k×2k 棋盘仅仅有一个残缺方格)。首先覆盖其中包含残缺方格的2k - 1×2k - 1 残缺棋盘,然后把剩下的3个小棋盘转变为残缺棋盘,为此将一个三格板放在由这3个小棋盘形成的角上,如图14-5b 所示,其中原2k×2k 棋盘中的残缺方格落入左上角的2k - 1×2k - 1 棋盘。可以采用这种分割技术递归地覆盖2k×2k 残缺棋盘。当棋盘的大小减为1×1时,递归过程终止。此时1×1的棋盘中仅仅包含一个方格且此方格残缺,所以无需放置三格板。. z+ E1 R' R4 O9 e; ~8 i3 v

* r) I# ^, j& p7 N0 F* @; B' Q可以将上述分而治之算法编写成一个递归的C++ 函数Ti l e B o a r d (见程序1 4 - 2 )。该函数定义了一个全局的二维整数数组变量B o a r d来表示棋盘。B o a r d [ 0 ] [ 0 ]表示棋盘中左上角的方格。该函数还定义了一个全局整数变量t i l e,其初始值为0。函数的输入参数如下:
0 M1 r: Q0 n4 Z
+ j) A: h5 C0 ^( x' @? tr 棋盘中左上角方格所在行。. p2 z, L! T& N1 E% O
, K: `( S: T9 x5 A+ y, j
? tc 棋盘中左上角方格所在列。
" I1 X4 N4 v# N, i3 l, Y8 d4 s: Y$ r/ ]! a7 O. i/ x7 j. Q0 }5 j
? dr 残缺方块所在行。
+ h. N+ C6 m9 r# H7 d- m5 T7 h/ f) `$ R) l. a" B
? dl 残缺方块所在列。/ G: A% e# p; U3 b/ z; [& v
% q9 a$ ~" ~* ~, J3 y6 z
? size 棋盘的行数或列数。
4 A: q( D" A9 F4 ^% Y5 ^4 i+ X* O  R! }0 n+ Q( l5 Q3 e2 M
Ti l e B o a r d函数的调用格式为Ti l e B o a r d(0,0, dr, dc,size),其中s i z e = 2k。覆盖残缺棋盘所需要的三格板数目为( s i z e2 -1 ) / 3。函数TileBoard 用整数1到( s i z e2-1 ) / 3来表示这些三格板,并用三格板的标号来标记被该三格板覆盖的非残缺方格。
, V3 f& o7 n! l$ Y. E. i& H
* L6 ]2 l7 Q1 s0 b9 y" B, E$ P- b令t (k) 为函数Ti l e B o a r d覆盖一个2k×2k 残缺棋盘所需要的时间。当k= 0时,s i z e等于1,覆盖它将花费常数时间d。当k &gt; 0时,将进行4次递归的函数调用,这些调用需花费的时间为4t (k-1 )。除了这些时间外, if 条件测试和覆盖3个非残缺方格也需要时间,假设用常数c 表示这些额外时间。可以得到以下递归表达式:( ^! B# T5 E( a
3 n$ j7 Y; j$ q& t; N
程序14-2 覆盖残缺棋盘8 d9 y/ ]9 ?9 ?. I6 \
0 U" v8 b0 C8 q  k" @
void TileBoard(int tr, int tc, int dr, int dc, int size)
2 U7 }" p0 T3 o8 [  f8 O$ o: w+ M' C& z$ D; f( G
{// 覆盖残缺棋盘  {2 ]* ~- X  O5 z/ y
3 J% o: [# z4 [: s2 r. [
if (size == 1) return;
: `% q3 s3 o' ]- d/ A7 B7 S1 r6 U" d7 h  j8 b
int t = tile++, // 所使用的三格板的数目
5 l; t. a2 p& B' f; Z
% Q: ~' v6 u4 Y  ?s = size/2; // 象限大小% x# L! B5 o0 F2 `; U

  t# n; t* M0 e% A8 z+ k) z! }/ /覆盖左上象限. Q5 q+ B3 K0 ]" G* t( y# g

4 f8 r# a" p  t" [if (dr &lt; tr + s &amp;&amp; dc &lt; tc + s)2 b3 J" a  K" }+ J' Q
9 b' d/ ]' U, |+ N# K$ q
// 残缺方格位于本象限
# R9 p  l. M( O7 O6 ]
3 H2 Q: Q' N: o0 W* |8 RTi l e B o a r d ( t r, tc, dr, dc, s);
/ @7 {: p5 G3 M# Q5 A3 O  U9 \" L3 e1 O: l- q
else {// 本象限中没有残缺方格( ~* ^$ y) L1 `7 f& J! b; u& _
. F- F$ {" j5 W: I2 m% Z
// 把三格板t 放在右下角
" S: m2 a* e2 H& Z4 c7 E9 E5 @; x
8 J2 \- N# D& S: {4 M# aBoard[tr + s - 1][tc + s - 1] = t;' s, m7 t1 @8 w0 K0 p" b

7 I1 u9 Q( h' ]2 Z3 Q4 _) x// 覆盖其余部分
0 z1 z! J9 m% t1 T
" A# `: W) V  p4 k$ ~. K  ZTi l e B o a r d ( t r, tc, tr+s-1, tc+s-1, s);}
! w. s. k$ M* ^5 w4 z# r, f5 E
1 i: `* B! G! J/ /覆盖右上象限
; z- e4 L9 G, h  n; m' @+ l9 ]4 b( {% e0 W: J' v7 Q
if (dr &lt; tr + s &amp;&amp; dc &gt;= tc + s)
1 D7 E: o( y) d% f
# i. w& A: _! t7 S0 f  @/ c// 残缺方格位于本象限! h0 _( x& G1 p- A3 ?+ R+ U) E
; e7 J  l$ |  o  {* v
Ti l e B o a r d ( t r, tc+s, dr, dc, s);
+ [- j, A# F# s$ z7 G; A4 v0 W  D
) ^5 W. i# @7 [8 a1 m( n/ ]else {// 本象限中没有残缺方格3 Z5 l7 F- e4 M" q$ E2 e$ b( N& A
! Z/ B+ t0 h& ^+ H
// 把三格板t 放在左下角7 C! n; j% I; G5 M1 f) [6 B; R
# w4 s7 D& m- j- q  S
Board[tr + s - 1][tc + s] = t;/ V( G5 B' h- M. f- Y# W

& [$ L- \2 E- G+ w- s// 覆盖其余部分
, U! C+ g0 x8 v, W: ]  _1 q8 n0 H/ ^: d2 f6 p( S( o$ l
Ti l e B o a r d ( t r, tc+s, tr+s-1, tc+s, s);}
8 C5 M' C- k7 ?5 ~0 v4 l) H3 a2 {& k5 v! J
/ /覆盖左下象限
$ O! f  V+ b% y9 e2 u7 e  o$ x# f( B) N6 p: g! z9 \! i. I, s
if (dr &gt;= tr + s &amp;&amp; dc &lt; tc + s)
4 z) O% w8 Z4 v& F0 ?0 s8 L% G( P$ j( g0 M0 F9 u" f9 C! A7 [
// 残缺方格位于本象限
7 \' {. z( A& g8 G6 B
( t" S4 l' p2 b: [" z( fTileBoard(tr+s, tc, dr, dc, s);: d" I7 V! v: `! i' I# m

( v( L& @5 j( @; h; A! telse {// 把三格板t 放在右上角
- P5 M7 v4 y, Q+ b
4 r. e/ `1 k' UBoard[tr + s][tc + s - 1] = t;
% U  L% ?, J0 }. V* ~
9 X. q$ p; I; K// 覆盖其余部分
( e( @* V6 G& E% _) R5 m' C% s+ C) n% |1 N/ B3 a8 T
TileBoard(tr+s, tc, tr+s, tc+s-1, s);}2 C( Y  k* W4 z% J
4 g' y4 E% w) K  k! s+ i
// 覆盖右下象限
9 Y& a3 E1 m( @+ P3 Q
( C, p+ |' {" ^' A$ I2 n! aif (dr &gt;= tr + s &amp;&amp; dc &gt;= tc + s)
* t: ~2 j6 F" V3 a+ X( M/ `# q* s/ `- h6 s0 o
// 残缺方格位于本象限. L% v* @; S0 W) ~5 ~6 Y( ?
: `' A% N2 f7 C6 y, p6 Q  b1 B
TileBoard(tr+s, tc+s, dr, dc, s);; D' j4 n9 f* i. J* c8 \4 e3 _$ |. C

0 A* Q. R3 V) J1 w- L5 n- y. oelse {// 把三格板t 放在左上角
3 C7 ^: a! D4 \9 x% P- ]8 i* f1 {  S* W* c5 ]
Board[tr + s][tc + s] = t;
  p) ~- _' ^0 A5 h6 E
8 G' \1 j# ]. W. X7 p// 覆盖其余部分# f; w# \8 x( D" i; Q+ Z4 e
1 [# U/ \$ E6 u- r$ [
TileBoard(tr+s, tc+s, tr+s, tc+s, s);}4 {- P  ]! G( S4 b! S$ k1 A  ?; t
2 S& j) e" A2 q, [6 X
}# G% K' @8 [! D' e( R
  s1 l5 k# H2 G0 N8 k
void OutputBoard(int size)
6 Y6 y3 Q/ K$ y& i" [
( }! Q) e4 y- o& N6 J, H8 j{
; Q, P2 k/ O$ E$ P  B1 j$ q! V0 j/ t) t% ~3 P7 G
for (int i = 0; i &lt; size; i++) {3 l$ i3 A1 n9 H" H. j

& l" R* k/ N5 V" kfor (int j = 0; j &lt; size; j++)
4 l  t% ?. m0 I& N. `8 F0 B* Q) B* ^
cout &lt;&lt; setw (5) &lt;&lt; Board[j];
; V& ?$ C& I0 `# x
4 H9 O2 z9 d) g6 C/ wcout &lt;&lt; endl;% }1 [. x8 K& i9 e
0 J4 G$ C+ W9 a. V1 C2 }# h6 X
}
1 J4 E4 K# [3 B* i  X5 t6 e
% `/ b3 x3 z0 P0 n, N( ?0 {}
% D" f. C# G' g( A3 \5 U  c" ], W1 |& x+ y: x- c- I
可以用迭代的方法来计算这个表达式(见例2 - 2 0),可得t (k )= ( 4k )= (所需的三格板的数目)。由于必须花费至少( 1 )的时间来放置每一块三格表,因此不可能得到一个比分而治之算法更快的算法。</P>
zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
yammay        

0

主题

0

听众

16

积分

升级  11.58%

该用户从未签到

新人进步奖

回复

使用道具 举报

wendy28        

0

主题

2

听众

24

积分

升级  20%

该用户从未签到

新人进步奖

回复

使用道具 举报

1

主题

2

听众

60

积分

升级  57.89%

该用户从未签到

新人进步奖

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2026-4-20 13:43 , Processed in 0.422768 second(s), 74 queries .

回顶部