Nickolay.info. Алгоритмы. Дамка и две шашки

В шашечном эндшпиле остались белая дамка и две чёрных пешки, позиции которых известны. Ход белых. Сможет ли дамка срубить одну или сразу обе пешки.

Вся суть задачи - как закодировать поля шахматной доски и как проверять, находится ли шашки на одной диагонали (ведь в шашках все ходы делаются по диагонали, как мы помним). Думаю, удобнее всего будет сделать так, как на рисунке:

задача о дамке и двух шашках

Кстати, изобразить такую доску быстрей всего в Excel формулой

=ЕСЛИ(ОСТАТ(B$10+$A2;2)=0;B$10*10+$A2;"")

- если доска (левый верхний угол доски) "начинается" с ячейки рабочего листа B2.

Алгоритмически решение несложно:

Пешка рубится, если она
 (Не на краю доски) И (Не рядом с другой пешкой) И
   ((На одной диагонали с дамкой) ИЛИ 
    (точка пересечения с диагональю другой пешки лежит по другую сторону от этой пешки, чем дамка;
     при этом другая пешка рубится напрямую))
Иначе не рубится

Функция ReadXY позволяет ввести допустимые координаты дамки и шашек, OnDiag проверяет, стоят ли 2 шашки на одной диагонали (в том числе, являются ли соседями), а CrossDiag вместе с DifferentSides отслеживают ходы "углом", когда дамка рубит сразу две шашки одним ходом. Можно, конечно, код оптимизировать, но здесь ставилась задача решить возможно быстрей и наглядней.

procedure Error (msg:string);
begin
 writeln;
 writeln (msg);
 write ('Нажмите ENTER для выхода из программы');
 reset (input); readln; halt;
end;

procedure ReadXY (msg:string; var x,y:integer);
var n:integer;
begin
 repeat
  writeln;
  write (msg);
  {$I-}read(n);{$I+}
  if (IoResult<>0) or (n<11) or (n>88) then begin
   writeln ('Недопустимый ввод! Введите координату клетки от 11 до 88');
   continue;
  end;
  x := n div 10;
  y := n mod 10;
  if (x>8) or (y>8) then begin
   writeln ('Ни одна из координат не может быть больше 8');
   continue;
  end;
  if (x+y) mod 2 = 1 then begin
   writeln ('Вы ввели координаты белой клетки, допустимы только черные');
   continue;
  end;
  break;
 until false;
end;

function OnEdge (x,y:integer):boolean;
begin
 if (x=1) or (x=8) or (y=1) or (y=8) then OnEdge:=true
 else OnEdge:=false;
end;

function Coord (x,y:integer):integer;
begin
 Coord:=x*10+y;
end;

function OnDiag (x1,y1,x2,y2:integer):integer;
 {Если шашки на одной диагонали, вернет расстояние в клетках
  между ними, иначе 0}
var n:integer;
begin
 n:=abs(Coord(x1,y1)-Coord(x2,y2));
 if n mod 11=0 then OnDiag:=n div 11
 else if n mod 9=0 then OnDiag:=n div 9
 else OnDiag:=0;
end;

function DifferentSides(x1,y1,x0,y0,x2,y2:integer):boolean;
{истина, если (x0,y0) лежит между остальными 2 точками}
var n1,n0,n2:integer;
begin
 n1:=Coord (x1,y1); n2:=Coord (x2,y2); n0:=Coord (x0,y0);
 if (n1<n0) and (n0<n2) or (n2<n0) and (n0<n1) then DifferentSides:=true
 else DifferentSides:=false;
end;

function CrossDiag(x1,y1,x2,y2,x0,y0:integer):boolean;
{Ищем точки пересечения (x,y) диагоналей шашек перебором -
 для точки пересечения характерно то, что она принадлежит
 диагоналям обеих шашек}
var x,y,k:integer;
begin
 if (OnDiag(x1,y1,x2,y2)>0) then begin
  {На одной диагноали не надо искать}
  CrossDiag:=false;
  Exit;
 end;
 x:=1; k:=0;
 while x<9 do begin
  if x mod 2=1 then y:=1 else y:=2;
  while y<9 do begin {Перебор всех черных клеток доски}
   if (OnDiag (x,y,x1,y1)>0) and (OnDiag (x,y,x2,y2)>0) then begin
    if DifferentSides(x,y,x1,y1,x0,y0)=true then begin
     CrossDiag:=true;
     Exit;
    end;
    inc(k);
    if k=2 then break; {Всего не более 2 точек пересечения}
   end;
   inc (y,2);
  end;
  inc(x);
 end;
 CrossDiag:=false;
end;

var x0,y0,x1,y1,x2,y2:integer;
    is1,is2:boolean;

begin
 {Вводим и проверяем координаты}
 ReadXY ('Координаты дамки: ',x0,y0);
 ReadXY ('Координаты 1-й шашки: ',x1,y1);
 ReadXY ('Координаты 2-й шашки: ',x2,y2);
 if (Coord(x0,y0)=Coord(x1,y1)) or (Coord(x0,y0)=Coord(x2,y2)) or
    (Coord(x1,y1)=Coord(x2,y2)) then
  Error ('Все 3 клетки должны быть разными!');
 is1:=(OnEdge(x1,y1)=false) and (OnDiag (x1,y1,x2,y2)<>1) and
      (OnDiag(x1,y1,x0,y0)>0);
 is2:=(OnEdge(x2,y2)=false) and (OnDiag (x1,y1,x2,y2)<>1) and
      (OnDiag(x2,y2,x0,y0)>0);
 write ('Шашка 1 ');
 if is1=true then writeln (' рубится напрямую')
 else if (is2=true) and (CrossDiag(x2,y2,x1,y1,x0,y0)=true)
  and (OnEdge(x1,y1)=false) then writeln (' рубится после второй')
 else writeln (' НЕ рубится');
 write ('Шашка 2 ');
 if is2=true then writeln (' рубится напрямую')
 else if (is1=true) and (CrossDiag(x1,y1,x2,y2,x0,y0)=true)
  and (OnEdge(x2,y2)=false) then writeln (' рубится после первой')
 else writeln (' НЕ рубится');
 writeln;
 write ('ENTER для выхода');
 reset (input); readln;
end.

Рейтинг@Mail.ru

вверх гостевая; E-mail