I found myself at a location without a phone, and introduced my boyfriend to Chopsticks, a game I played in elementary school in Korea. (It appears to be Japanese in origin.) It’s a simple game, and I napkin-calculated it could have at most* 450 states.
Then I became curious what the full game state transition diagram would look like.
Blue is start. Green and red are starting player and nonstarting player win nodes.
As far as I can tell, the actual number of reachable states is 382, counting the 28 arguably-not-valid(-distinct) game states where loser is out of hands and the winner has one of 14 different hand states (01, 02, 03, 04, 12, 13, 14, 22, 23, 24, 33, 34, 44).
The diagram skips those 14 states – if a player can win in a game state, that state immediately goes to the player’s win node.
Python code below:
from collections import deque
from graph_tool.all import *
start = ((1, 1), (1, 1), 0) # p1hands, p2hands, turn. if turn is 0, p1 goes next. phands are always sorted
end = [2, 1] # p2 wins, p1 wins
q = deque([start])
mapping = {start: []}
# returns a deduped set of [(hand1, hand2), (hand1, hand2)]. hands will be sorted.
def getNextHands(activePlayer, passivePlayer): # tuple, tuple
a, b = activePlayer
c, d = passivePlayer
moves = set()
if a+b > 1 and a+b < 7:
for i in range(max(0, a+b-4), min(5, a+b)):
moves.add((
tuple(sorted((i, a+b-i))),
passivePlayer
))
moves.discard((activePlayer, passivePlayer)) # discard state identical the one we started with
if a > 0:
if c > 0:
moves.add((activePlayer, tuple(sorted((c+a if c+a < 5 else 0, d)))))
if d > 0:
moves.add((activePlayer, tuple(sorted((c, d+a if d+a < 5 else 0)))))
if b > 0:
if c > 0:
moves.add((activePlayer, tuple(sorted((c+b if c+b < 5 else 0, d)))))
if d > 0:
moves.add((activePlayer, tuple(sorted((c, d+b if d+b < 5 else 0)))))
return moves
# state => [state, ... state]
def getNextStates(state):
if state[-1] == 0: # active player is first player
hands = getNextHands(state[0], state[1])
return [(h + (1,))for h in hands]
else: # active player is second
hands = getNextHands(state[1], state[0])
return [(h[1], h[0], 0,) for h in hands]
def buildMapping():
global q
global mapping
while len(q) > 0:
gameState = q.popleft()
if (0, 0) in gameState: # lose state; no successors
mapping[gameState] = [end[gameState[-1]]]
else:
successors = getNextStates(gameState)
mapping[gameState]= successors
q.extend(filter(lambda s: s not in mapping, successors))
# graph tool wants nodes that are labeled 0 to N, where N is how many vertices there are. if there's a gap, it'll create an isolated vertex, so we need to remove gaps
def reduceNodes(m1):
keymap = {start: 0, 1: 1, 2: 2}
nextNewKey = 3
for k in m1:
if k not in keymap:
keymap[k] = nextNewKey
nextNewKey += 1
reducedMap = {}
for k, vs in m1.items():
reducedMap[keymap[k]] = list(map(lambda x: keymap[x], vs))
return reducedMap
buildMapping()
rekeyed = reduceNodes(mapping)
g = Graph(rekeyed)
vcolor = g.new_vertex_property("vector<double>")
for v in g.vertices():
vcolor[v] = [1, 0.2, 0.2, 0.5]
vcolor[g.vertex(0)] = [1, 0, 0, 1]
vcolor[g.vertex(1)] = [0, 0.7, 1, 1]
vcolor[g.vertex(2)] = [0, 0.7, 1, 1]
pos = sfdp_layout(g, K=1.7)
graph_draw(g, pos=pos, vertex_fill_color=vcolor)