Python 2 3、141-15 = 126
def win(x,y):w([y]*x)
w=lambda b,f=print:not[f(r+1,c+1)for r,p in enumerate(b)for c in range(p)if(r+c)*w(b[:r]+[min(i,c)for i in b[r:]],max)]
ブルートフォースミニマックス検索。考えられるすべての動きについて、その動きをした後に対戦相手が勝つことができるかどうかを再帰的に確認します。かなり弱いゴルフ。他の誰かがもっとうまくやれるはずです。これはAPLの仕事のように感じます。
winパブリックインターフェイスです。ボードの寸法を取得し、ボード表現に変換し、それをに渡しwます。
wはミニマックスアルゴリズムです。ボードの状態を取得し、すべての動きを試み、勝った動きに対応する要素を持つリストを作成し、リストが空の場合にTrueを返します。デフォルトf=printでは、リストを作成すると、勝ち手が印刷されるという副作用があります。関数名は勝った動きのリストを返したときにもっと意味をなしていたがnot、スペースを節約するためにリストの前に移動した。
for r,p in enumerate(b)for c in xrange(p) if(r+c):考えられるすべての動きを繰り返します。 1 1は、基本的なケースを少し単純化して、合法的な動きではないものとして扱われます。
b[:r]+[min(i,c)for i in b[r:]]:座標rで表される移動後のボードの状態を構築し、cます。
w(b[:r]+[min(i,c)for i in b[r:]],max):新しい状態が負け状態であるかどうかを確認するために再帰します。 maxは、2つの整数引数を取り、文句を言わない最短の関数です。
f(r+1,c+1):f印刷の場合、移動を印刷します。なんでもf、リストの長さを埋めるための値を生成します。
not [...]:空のリストと空でないリストをnot返します。TrueFalse
はるかに大きな入力を処理するメモ化を含む、完全に未使用のオリジナルのPython 2コード:
def win(x, y):
for row, column in _win(Board([y]*x)):
print row+1, column+1
class MemoDict(dict):
def __init__(self, func):
self.memofunc = func
def __missing__(self, key):
self[key] = retval = self.memofunc(key)
return retval
def memoize(func):
return MemoDict(func).__getitem__
def _normalize(state):
state = tuple(state)
if 0 in state:
state = state[:state.index(0)]
return state
class Board(object):
def __init__(self, state):
self.state = _normalize(state)
def __eq__(self, other):
if not isinstance(other, Board):
return NotImplemented
return self.state == other.state
def __hash__(self):
return hash(self.state)
def after(self, move):
row, column = move
newstate = list(self.state)
for i in xrange(row, len(newstate)):
newstate[i] = min(newstate[i], column)
return Board(newstate)
def moves(self):
for row, pieces in enumerate(self.state):
for column in xrange(pieces):
if (row, column) != (0, 0):
yield row, column
def lost(self):
return self.state == (1,)
@memoize
def _win(board):
return [move for move in board.moves() if not _win(board.after(move))]
デモ:
>>> for i in xrange(7, 11):
... for j in xrange(7, 11):
... print 'Dimensions: {} by {}'.format(i, j)
... win(i, j)
...
Dimensions: 7 by 7
2 2
Dimensions: 7 by 8
3 3
Dimensions: 7 by 9
3 4
Dimensions: 7 by 10
2 3
Dimensions: 8 by 7
3 3
Dimensions: 8 by 8
2 2
Dimensions: 8 by 9
6 7
Dimensions: 8 by 10
4 9
5 6
Dimensions: 9 by 7
4 3
Dimensions: 9 by 8
7 6
Dimensions: 9 by 9
2 2
Dimensions: 9 by 10
7 8
9 5
Dimensions: 10 by 7
3 2
Dimensions: 10 by 8
6 5
9 4
Dimensions: 10 by 9
5 9
8 7
Dimensions: 10 by 10
2 2