Skip to content

Latest commit

 

History

History
268 lines (212 loc) · 10.5 KB

File metadata and controls

268 lines (212 loc) · 10.5 KB

Basic Reinforcement Learning Tutorial 1: Q-learning

This tutorial covers the basics of Q-learning using the cat-mouse-cheese example.

Table of Contents

### Background

Value Functions are state-action pair functions that estimate how good a particular action will be in a given state, or what the return for that action is expected to be.

Q-Learning is an off-policy (can update the estimated value functions using hypothetical actions, those which have not actually been tried) algorithm for temporal difference learning ( method to estimate value functions). It can be proven that given sufficient training, the Q-learning converges with probability 1 to a close approximation of the action-value function for an arbitrary target policy. Q-Learning learns the optimal policy even when actions are selected according to a more exploratory or even random policy. Q-learning can be implemented as follows:

Initialize Q(s,a) arbitrarily
Repeat (for each generation):
	Initialize state s
	While (s is not a terminal state):		
		Choose a from s using policy derived from Q
		Take action a, observe r, s'
		Q(s,a) += alpha * (r + gamma * max,Q(s') - Q(s,a))
		s = s'

where:

  • s: is the previous state
  • a: is the previous action
  • Q(): is the Q-learning algorithm
  • s': is the current state
  • alpha: is the the learning rate, set generally between 0 and 1. Setting it to 0 means that the Q-values are never updated, thereby nothing is learned. Setting alpha to a high value such as 0.9 means that learning can occur quickly.
  • gamma: is the discount factor, also set between 0 and 1. This models the fact that future rewards are worth less than immediate rewards.
  • max,: is the the maximum reward that is attainable in the state following the current one (the reward for taking the optimal action thereafter).

The algorithm can be interpreted as:

  1. Initialize the Q-values table, Q(s, a).
  2. Observe the current state, s.
  3. Choose an action, a, for that state based on the selection policy.
  4. Take the action, and observe the reward, r, as well as the new state, s'.
  5. Update the Q-value for the state using the observed reward and the maximum reward possible for the next state.
  6. Set the state to the new state, and repeat the process until a terminal state is reached.
### The *World* and *Cat* player implementations

The implementations of the discrete 2D world (including agents, cells and other abstractions) as well as the cat and mouse players is performed in the cellular.py file. The world is generated from a .txtfile. In particular, I'm using the worlds/waco.txt:

(waco world)

XXXXXXXXXXXXXX
X            X
X XXX X   XX X
X  X  XX XXX X
X XX      X  X
X    X  X    X
X X XXX X XXXX
X X  X  X    X
X XX   XXX  XX
X    X       X
XXXXXXXXXXXXXX

The Cat player class inherit from cellular.Agent and its implementation is set to follow the Mouse player:

    def goTowards(self, target):
        if self.cell == target:
            return
        best = None
        for n in self.cell.neighbours:
            if n == target:
                best = target
                break
            dist = (n.x - target.x) ** 2 + (n.y - target.y) ** 2
            if best is None or bestDist > dist:
                best = n
                bestDist = dist
        if best is not None:
            if getattr(best, 'wall', False):
                return
            self.cell = best

The Cat player calculates the quadratic distance (bestDist) among its neighbours and moves itself (self.cell = best) to that cell.

class Cat(cellular.Agent):
    cell = None
    score = 0
    colour = 'orange'

    def update(self):
        cell = self.cell
        if cell != mouse.cell:
            self.goTowards(mouse.cell)
            while cell == self.cell:
                self.goInDirection(random.randrange(directions))

Overall, the Cat pursues the Mouse through the goTowards method by calculating the quadratic distance. Whenever it bumps to the wall, it takes a random action.

### The *Mouse* player

The Mouse player contains the following attributes:

class Mouse(cellular.Agent):
    colour = 'gray'

    def __init__(self):
        self.ai = None
        self.ai = qlearn.QLearn(actions=range(directions),
                                alpha=0.1, gamma=0.9, epsilon=0.1)
        self.eaten = 0
        self.fed = 0
        self.lastState = None
        self.lastAction = None

The eaten and fed attributes store the performance of the player while the lastState and lastAction ones help the Mouse player store information about its previous states (later used for learning).

The ai attribute stores the Q-learning implementation which is initialized with the following parameters:

  • directions: There're different possible directions/actions implemented at the getPointInDirection method of the World class:
    def getPointInDirection(self, x, y, dir):
        if self.directions == 8:
            dx, dy = [(0, -1), (1, -1), (
                1, 0), (1, 1), (0, 1), (-1, 1), (-1, 0), (-1, -1)][dir]
        elif self.directions == 4:
            dx, dy = [(0, -1), (1, 0), (0, 1), (-1, 0)][dir]
        elif self.directions == 6:
            if y % 2 == 0:
                dx, dy = [(1, 0), (0, 1), (-1, 1), (-1, 0),
                          (-1, -1), (0, -1)][dir]
            else:
                dx, dy = [(1, 0), (1, 1), (0, 1), (-1, 0),
                          (0, -1), (1, -1)][dir]

In general, this implementaiton will be used in 8 directions, thererby (0, -1), (1, -1), (1, 0), (1, 1), (0, 1), (-1, 1), (-1, 0), (-1, -1)].

  • alpha: which is the q-learning discount constant, set to 0.1.
  • gamma: the q-learning discount factor, set to 0.9.
  • epsilon: an exploration constant to randomize decisions, set to 0.1.

The Mouse player calculates the next state using the calcState() method implemented as follows:

    def calcState(self):
        def cellvalue(cell):
            if cat.cell is not None and (cell.x == cat.cell.x and
                                         cell.y == cat.cell.y):
                return 3
            elif cheese.cell is not None and (cell.x == cheese.cell.x and
                                              cell.y == cheese.cell.y):
                return 2
            else:
                return 1 if cell.wall else 0

        return tuple([cellvalue(self.world.getWrappedCell(self.cell.x + j, self.cell.y + i))
                      for i,j in lookcells])

This, in a nutshell, returns a tupple of the values of the cells surrounding the current Mouse as follows:

  • 3: if the Cat is in that cell
  • 2: if the Cheese is in that cell
  • 1: if the that cell is a wall
  • 0: otherwise

The lookup is performed according to the lookdist variable that in this implementation uses a value of 2 (in other words, the mouse can "see" up to two cells ahead in every direction).

To finish up reviewing the Mouse implementation, let's look at how the Q-learning is implemented:

#### Q-learning implementation Let's look at the `update` method of the *Mouse* player:
    def update(self):
        # calculate the state of the surrounding cells
        state = self.calcState()
        # asign a reward of -1 by default
        reward = -1

        # Update the Q-value
        if self.cell == cat.cell:
            self.eaten += 1
            reward = -100
            if self.lastState is not None:
                self.ai.learn(self.lastState, self.lastAction, reward, state)
            self.lastState = None

            self.cell = pickRandomLocation()
            return

        if self.cell == cheese.cell:
            self.fed += 1
            reward = 50
            cheese.cell = pickRandomLocation()

        if self.lastState is not None:
            self.ai.learn(self.lastState, self.lastAction, reward, state)

        # Choose a new action and execute it
        state = self.calcState()
        action = self.ai.chooseAction(state)
        self.lastState = state
        self.lastAction = action

        self.goInDirection(action)

Code has been commented so that its understanding is simplified. The implementation matches the pseudo-code presented in the Background section above (note that for the sake of the implementation, the actions in the Python implementation have been reordered).

Rewards are given with these terms:

  • -100: if the Cat player eats the Mouse
  • 50: if the Mouse player eats the cheese
  • -1: otherwise

The learning algorithm records every state/action/reward combination in a dictionary containing a (state, action) tuple in the key and the reward as the value of each member.

Note that the amount of elements saved in the dictionary for this simplified 2D environment is considerable after a few generations. To get some insight about this fact, consider the following numbers:

  • After 10 000 generations:
    • 2430 elements (state/action/reward combinations) learned
    • Bytes: 196888 (192 KB)
  • After 100 000 generations:
    • 5631 elements (state/action/reward combinations) learned
    • Bytes: 786712 (768 KB)
  • After 600 000 generations:
    • 9514 elements (state/action/reward combinations) learned
    • Bytes: 786712 (768 KB)
  • After 1 000 000 generations:
    • 10440 elements (state/action/reward combinations) learned
    • Bytes: 786712 (768 KB)

Given the results showed above, one can observe that for some reason, Python sys.getsizeof function seems to be upper bounded by 786712 (768 KB). We can't provide accurate data but given the results showed, one can conclude that the elements generated after 1 million generations should require something close to 10 MB in memory for this 2D simplified world.

### Results Below we present a *mouse player* after **15 000 generations** of reinforcement learning:

and now we present the same player after 150 000 generations of reinforcement learning:

### Reproduce it yourself
git clone https://github.com/vmayoral/basic_reinforcement_learning
cd basic_reinforcement_learning
python tutorial1/egoMouseLook.py