COMP343 - Tic-Tac-Toe Agents
Agents
For an agent to leverage the capabilities of TTTEngine
it must provide two capabilities:
- A read-only
marker
property which reports'x'
or'o'
depending on the agent’s mark. - A method
getMove
which takes in a board state (aTTTState
object) and returns the row and column coordinates for the location in which the player wishes to make their mark.
In a nutshell, agents need to be able to report on which mark their playing and where they want to play. That’s it.
Agent
and AbstractAgent
Base Classes
The above requirements for an agent are captured and made explicit by the class AbstractAgent
. This is an abstract base class; it provides no concrete implementation for agents but will hold any class that extends it to the required interface for an agent.
The AbstractAgent
class is extended by the class Agent
this class fully implements all the needed functionality for the marker
property. Thus, any class that extends Agent
need only provide a concrete implementation for getMove
.
This means agent designers like you need only extend the Agent
class and provide an implementation for the following method:
def getMove(self,st : TTTState) -> Tuple[int,int]:
"""
Given the current game state, return the desired move. Moves are specified by
the a (row,col) tuple where 0 <=row,col< 3. The move is assumed to be legal.
Parameters:
st : TTTState - the current state of the game
Return:
Tuple[int,int] - return the desired move
"""
Let’s begin by looking at Random
, a not-so-intelligent agent build by extending Agent
.
Random
Agent and Implementing Your Own Agent
A Random
agent will make a random selection from all available moves. This is not rational, but it is agent-like behavior. The following definition for Random
can be found in the agent
module.
class Random(Agent):
"""
Random agent. Agent will choose an available move at random.
"""
def __init__(self, mark : str):
super().__init__(mark)
def getMove(self,st : TTTState) -> Tuple[int,int]:
"""
Agent chooses a random, legal move.
Parameters:
st : TTTState - current board state
Return:
Tuple[int,int] - row and column coordinates of the agents move.
"""
op = list(st.getOpen())
return random.choice(op)
First, note that the class extends the Agent
class via the declaration class Random(Agent)
. This means that it will inherit all the concrete code implemented within Agent
as well as the obligation to provide a concrete implementation for getMove
that Agent
inherited when it extended AbstractAgent
.
To successfully leverage Agent
’s marker
property implementation, the Random
agent invokes the Agent
constructor via super().__init__(mark)
within its own constructor. So long as your agent class repeats this pattern, it will also inherit all of the maker property logic.
Finally, Random
provides an implementation for getMove
with the exact signature specified by AbstractAgent
. In doing so, it complete’s the general agent interface and can be integrated with the tic-tac-toe logic provided by the tic-tac-toe modules.
Human
Agent
The Human
agent extends Agent
and allows for a human to play tic-tac-toe using the provided tic-tac-toe logic. You can explore the code for this agent in the agent
module. Here we’ll just take note that the header and constructor follow the same pattern seen in the Random
class: it extends Agent
and invokes Agent
’s construct as part of it’s own constructor.
class Human(Agent):
"""
Human tic-tac-toe player. Moves are determined via interactive text input.
"""
def __init__(self, mark : str):
"""
Construct human agent to play as given mark.
Parameters:
mark : str - 'x' or 'o'
"""
super().__init__(mark)
The getMove
implementation for Human
is interactive. It requests, as text-input, the move from the Human. Human agents must express the row and column coordinates on a 1 to 3 scale rather than the CS standard 0 to 2 scale. Human agent provided input is checked and validated to ensure that an allowable move is selected. One notable
A Template for Building Agents
All of these leads to a template of sorts for building agents.
class YOURAGENT(Agent):
def __init__(self,mark):
super().__init__(mark)
...
def getMove(self,board):
...
From here you can add more attributes to your agent and initialize them in the constructor. You can also add additional methods to the agent. Just treat getMove
as the main method when it comes time to play tic-tac-toe. The TTTEngine
that manages the game will call out to this method (and the marker
) property and nothing else.