###############################################################################
## TSP SOLUTION WITH GENETIC ALGORITHM ##
## Luke Collins ~ ICS2207 Assignment ##
###############################################################################
import sys, math, random
import numpy as np
import matplotlib.pyplot as plt
class Graph:
"""Graph data structure"""
def __init__(self, vertices):
self.vertices = vertices
self.n = len(vertices)
# Weighted adjacency matrix
self.a = np.empty((self.n, self.n))
self.a[:] = np.nan
def d(self, i, j):
"""Euclidean Metric d_2((x1, y1), (x2, y2))"""
# Check if the distance was computed before
if not np.isnan(self.a[i][j]):
return self.a[i][j]
_x1 = self.vertices[i][0]
_x2 = self.vertices[j][0]
_y1 = self.vertices[i][1]
_y2 = self.vertices[j][1]
# Otherwise compute it
_distance
= math.
sqrt((_x1
- _x2
)**2 + (_y1
- _y2
)**2)
# Add to matrix
self.a[i][j], self.a[j][i] = _distance, _distance
return _distance
def plot(self, tour=None):
"""Plots the cities and superimposes given tour"""
if tour is None:
tour = Tour(self, [])
_vertices = [self.vertices[0]]
for i in tour.vertices:
_vertices.append(self.vertices[i])
_vertices.append(self.vertices[0])
plt.title("Cost = " + str(tour.cost()))
plt.plot(*zip(*_vertices), '-r')
plt.scatter(*zip(*self.vertices), c="b", s=10, marker="s")
plt.show()
class Tour:
"""Tour data structure"""
def __init__(self, g, vertices = None):
"""Generate random tour in given graph g"""
self.g = g
if vertices is None:
self.vertices = list(range(1, g.n))
random.shuffle(self.vertices)
else:
self.vertices = vertices
self.__cost = None
def cost(self):
"""Return total edge-cost of tour"""
if self.__cost is None:
self.__cost = self.g.d(0, self.vertices[0]) + \
sum(map(self.g.d, self.vertices, self.vertices[1:])) + \
self.g.d(self.vertices[-1], 0)
return self.__cost
class GeneticAlgorithm:
"""Implements instance of genetic algorithm"""
def __init__(self, g, population_size, k=3, elite_mating_rate=0.3,
mutation_rate=0.1, mutation_swap_rate=0.5,
random_least_fit=0.1):
"""Initialises algorithm parameters"""
self.g = g
self.n = g.n
self.population = []
for _ in range(population_size):
self.population.append(Tour(g))
self.population_size = population_size
self.k = k
self.elite_mating_rate = elite_mating_rate
self.mutation_rate = mutation_rate
self.mutation_swap_rate = mutation_swap_rate
self.random_least_fit = random_least_fit
def crossover(self, mum, dad):
"""Implements ordered crossover"""
size = self.n - 1
# Choose random start/end position for crossover
start, end = sorted([random.randrange(size) for _ in range(2)])
# Identify the elements from mum's sequence which end up in alice,
# and from dad's which end up in bob
mumxo = set(mum.vertices[start:end + 1])
dadxo = set(dad.vertices[start:end + 1])
# Take the other elements in their original order
alice = [i for i in dad.vertices if not i in mumxo]
bob = [i for i in mum.vertices if not i in dadxo]
# Insert selected elements of mum's sequence for alice, dad's for bob
alice[start:start] = mum.vertices[start:end + 1]
bob[start:start] = dad.vertices[start:end + 1]
# Return twins
return Tour(self.g, alice), Tour(self.g, bob)
def mutate(self, tour):
"""Randomly swaps pairs of cities in a given tour according to mutation rate"""
# Decide whether to mutate
if random.random() < self.mutation_rate:
# For each vertex
for i in range(self.n - 1):
# Randomly decide whether to swap
if random.random() < self.mutation_swap_rate:
# Randomly choose other city position
j = random.randrange(self.n - 1)
# Swap
tour.vertices[i], tour.vertices[j] = tour.vertices[j], tour.vertices[i]
def select_parent(self, k):
"""Implements k-tournament selection to choose parents"""
tournament = random.sample(self.population, k)
return max(tournament, key=lambda t: t.cost())
def evolve(self):
"""Executes one iteration of the genetic algorithm to obtain a new generation"""
new_population = []
for _ in range(self.population_size):
# K-tournament for parents
mum, dad = self.select_parent(self.k), self.select_parent(self.k)
alice, bob = self.crossover(mum, dad)
# Mate in an elite fashion according to the elitism_rate
if random.random() < self.elite_mating_rate:
if alice.cost() < mum.cost() or alice.cost() < dad.cost():
new_population.append(alice)
if bob.cost() < mum.cost() or bob.cost() < dad.cost():
new_population.append(bob)
else:
self.mutate(alice)
self.mutate(bob)
new_population += [alice, bob]
# Add new population to old
self.population += new_population
# Retain fittest
self.population = sorted(self.population, key=lambda t: t.cost())[0:self.population_size]
def run(self, iterations):
"""Runs the GA until a pair of consecutive tours differ in cost by epsilon"""
for _ in range(iterations):
self.evolve()
# Replace least fit of population and replace with new random generated
# tours
retain = int((1-self.random_least_fit) * self.population_size)
to_add = self.population_size - retain
self.population = self.population[0:retain]
for _ in range(to_add):
self.population.append(Tour(self.g))
def best(self):
return min(self.population, key=lambda t: t.cost())
# Test on berlin52: http://e...content-available-to-author-only...b.de/pub/mp-testdata/tsp/tsplib/tsp/berlin52.tsp
g = Graph([(565.0, 575.0), (25.0, 185.0), (345.0, 750.0), (945.0, 685.0),
(845.0, 655.0), (880.0, 660.0), (25.0, 230.0), (525.0, 1000.0),
(580.0, 1175.0), (650.0, 1130.0), (1605.0, 620.0), (1220.0, 580.0),
(1465.0, 200.0), (1530.0, 5.0), (845.0, 680.0), (725.0, 370.0),
(145.0, 665.0), (415.0, 635.0), (510.0, 875.0), (560.0, 365.0),
(300.0, 465.0), (520.0, 585.0), (480.0, 415.0), (835.0, 625.0),
(975.0, 580.0), (1215.0, 245.0), (1320.0, 315.0), (1250.0, 400.0),
(660.0, 180.0), (410.0, 250.0), (420.0, 555.0), (575.0, 665.0),
(1150.0, 1160.0), (700.0, 580.0), (685.0, 595.0), (685.0, 610.0),
(770.0, 610.0), (795.0, 645.0), (720.0, 635.0), (760.0, 650.0),
(475.0, 960.0), (95.0, 260.0), (875.0, 920.0), (700.0, 500.0),
(555.0, 815.0), (830.0, 485.0), (1170.0, 65.0), (830.0, 610.0),
(605.0, 625.0), (595.0, 360.0), (1340.0, 725.0), (1740.0, 245.0)])
# Population size 200, 8000 iterations
ga = GeneticAlgorithm(g, 200)
ga.run(8000)
best_tour = ga.best()
g.plot(best_tour)
###############################################################################
##                   TSP SOLUTION WITH GENETIC ALGORITHM                     ##
##                    Luke Collins ~ ICS2207 Assignment                      ##
###############################################################################

import sys, math, random
import numpy as np
import matplotlib.pyplot as plt

class Graph:
    """Graph data structure"""

    def __init__(self, vertices):
        self.vertices = vertices
        self.n = len(vertices)

        # Weighted adjacency matrix
        self.a = np.empty((self.n, self.n))
        self.a[:] = np.nan

    def d(self, i, j):
        """Euclidean Metric d_2((x1, y1), (x2, y2))"""

        # Check if the distance was computed before
        if not np.isnan(self.a[i][j]):
            return self.a[i][j]

        _x1 = self.vertices[i][0]
        _x2 = self.vertices[j][0]
        _y1 = self.vertices[i][1]
        _y2 = self.vertices[j][1]

        # Otherwise compute it
        _distance = math.sqrt((_x1 - _x2)**2 + (_y1 - _y2)**2)

        # Add to matrix
        self.a[i][j], self.a[j][i] = _distance, _distance
        return _distance

    def plot(self, tour=None):
        """Plots the cities and superimposes given tour"""

        if tour is None:
            tour = Tour(self, [])

        _vertices = [self.vertices[0]]

        for i in tour.vertices:
            _vertices.append(self.vertices[i])

        _vertices.append(self.vertices[0])

        plt.title("Cost = " + str(tour.cost()))
        plt.plot(*zip(*_vertices), '-r')
        plt.scatter(*zip(*self.vertices), c="b", s=10, marker="s")
        plt.show()

class Tour:
    """Tour data structure"""

    def __init__(self, g, vertices = None):
        """Generate random tour in given graph g"""

        self.g = g

        if vertices is None:
            self.vertices = list(range(1, g.n))
            random.shuffle(self.vertices)
        else:
            self.vertices = vertices

        self.__cost = None

    def cost(self):
        """Return total edge-cost of tour"""
        if self.__cost is None:
            self.__cost = self.g.d(0, self.vertices[0]) + \
                          sum(map(self.g.d, self.vertices, self.vertices[1:])) + \
                          self.g.d(self.vertices[-1], 0)
        return self.__cost


class GeneticAlgorithm:
    """Implements instance of genetic algorithm"""

    def __init__(self, g, population_size, k=3, elite_mating_rate=0.3,
                 mutation_rate=0.1, mutation_swap_rate=0.5,
                 random_least_fit=0.1):
        """Initialises algorithm parameters"""

        self.g = g
        self.n = g.n

        self.population = []
        for _ in range(population_size):
            self.population.append(Tour(g))

        self.population_size = population_size
        self.k = k
        self.elite_mating_rate = elite_mating_rate
        self.mutation_rate = mutation_rate
        self.mutation_swap_rate = mutation_swap_rate
        self.random_least_fit = random_least_fit

    def crossover(self, mum, dad):
        """Implements ordered crossover"""

        size = self.n - 1

        # Choose random start/end position for crossover
        start, end = sorted([random.randrange(size) for _ in range(2)])

        # Identify the elements from mum's sequence which end up in alice,
        # and from dad's which end up in bob
        mumxo = set(mum.vertices[start:end + 1])
        dadxo = set(dad.vertices[start:end + 1])

        # Take the other elements in their original order
        alice = [i for i in dad.vertices if not i in mumxo]
        bob = [i for i in mum.vertices if not i in dadxo]

        # Insert selected elements of mum's sequence for alice, dad's for bob
        alice[start:start] = mum.vertices[start:end + 1]
        bob[start:start] = dad.vertices[start:end + 1]

        # Return twins
        return Tour(self.g, alice), Tour(self.g, bob)

    def mutate(self, tour):
        """Randomly swaps pairs of cities in a given tour according to mutation rate"""

        # Decide whether to mutate
        if random.random() < self.mutation_rate:

            # For each vertex
            for i in range(self.n - 1):

                # Randomly decide whether to swap
                if random.random() < self.mutation_swap_rate:

                    # Randomly choose other city position
                    j = random.randrange(self.n - 1)

                    # Swap
                    tour.vertices[i], tour.vertices[j] = tour.vertices[j], tour.vertices[i]

    def select_parent(self, k):
        """Implements k-tournament selection to choose parents"""
        tournament = random.sample(self.population, k)
        return max(tournament, key=lambda t: t.cost())

    def evolve(self):
        """Executes one iteration of the genetic algorithm to obtain a new generation"""

        new_population = []

        for _ in range(self.population_size):

            # K-tournament for parents
            mum, dad = self.select_parent(self.k), self.select_parent(self.k)
            alice, bob = self.crossover(mum, dad)

            # Mate in an elite fashion according to the elitism_rate
            if random.random() < self.elite_mating_rate:
                if alice.cost() < mum.cost() or alice.cost() < dad.cost():
                    new_population.append(alice)
                if bob.cost() < mum.cost() or bob.cost() < dad.cost():
                    new_population.append(bob)

            else:
                self.mutate(alice)
                self.mutate(bob)
                new_population += [alice, bob]

        # Add new population to old
        self.population += new_population

        # Retain fittest
        self.population = sorted(self.population, key=lambda t: t.cost())[0:self.population_size]

    def run(self, iterations):
        """Runs the GA until a pair of consecutive tours differ in cost by epsilon"""
        for _ in range(iterations):
            self.evolve()

            # Replace least fit of population and replace with new random generated
            # tours
            retain = int((1-self.random_least_fit) * self.population_size)
            to_add = self.population_size - retain
            self.population = self.population[0:retain]
            for _ in range(to_add):
                self.population.append(Tour(self.g))

    def best(self):
        return min(self.population, key=lambda t: t.cost())

# Test on berlin52: http://e...content-available-to-author-only...b.de/pub/mp-testdata/tsp/tsplib/tsp/berlin52.tsp
g = Graph([(565.0, 575.0), (25.0, 185.0), (345.0, 750.0), (945.0, 685.0), 
           (845.0, 655.0), (880.0, 660.0), (25.0, 230.0), (525.0, 1000.0), 
           (580.0, 1175.0), (650.0, 1130.0), (1605.0, 620.0), (1220.0, 580.0), 
           (1465.0, 200.0), (1530.0, 5.0), (845.0, 680.0), (725.0, 370.0), 
           (145.0, 665.0), (415.0, 635.0), (510.0, 875.0), (560.0, 365.0), 
           (300.0, 465.0), (520.0, 585.0), (480.0, 415.0), (835.0, 625.0), 
           (975.0, 580.0), (1215.0, 245.0), (1320.0, 315.0), (1250.0, 400.0), 
           (660.0, 180.0), (410.0, 250.0), (420.0, 555.0), (575.0, 665.0), 
           (1150.0, 1160.0), (700.0, 580.0), (685.0, 595.0), (685.0, 610.0), 
           (770.0, 610.0), (795.0, 645.0), (720.0, 635.0), (760.0, 650.0), 
           (475.0, 960.0), (95.0, 260.0), (875.0, 920.0), (700.0, 500.0), 
           (555.0, 815.0), (830.0, 485.0), (1170.0, 65.0), (830.0, 610.0), 
           (605.0, 625.0), (595.0, 360.0), (1340.0, 725.0), (1740.0, 245.0)])

# Population size 200, 8000 iterations
ga = GeneticAlgorithm(g, 200)
ga.run(8000)

best_tour = ga.best()
g.plot(best_tour)