#!/usr/bin/env python # coding: utf-8 # # Table of Contents #
# # Simulated annealing in Python # # This small notebook implements, in [Python 3](https://docs.python.org/3/), the [simulated annealing](https://en.wikipedia.org/wiki/Simulated_annealing) algorithm for numerical optimization. # # ## References # - The Wikipedia page: [simulated annealing](https://en.wikipedia.org/wiki/Simulated_annealing). # - It was implemented in `scipy.optimize` before version 0.14: [`scipy.optimize.anneal`](https://docs.scipy.org/doc/scipy-0.15.1/reference/generated/scipy.optimize.anneal.html). # - [This blog post](http://apmonitor.com/me575/index.php/Main/SimulatedAnnealing). # - These Stack Overflow questions: [15853513](https://stackoverflow.com/questions/15853513/) and [19757551](https://stackoverflow.com/questions/19757551/). # # ## See also # - For a real-world use of simulated annealing, this Python module seems useful: [perrygeo/simanneal on GitHub](https://github.com/perrygeo/simanneal). # # ## About # - *Date:* 20/07/2017. # - *Author:* [Lilian Besson](https://GitHub.com/Naereen), (C) 2017. # - *Licence:* [MIT Licence](http://lbesson.mit-license.org). # # ---- # > This notebook should be compatible with both Python versions, [2](https://docs.python.org/2/) and [3](https://docs.python.org/3/). # In[1]: from __future__ import print_function, division # Python 2 compatibility if needed # In[2]: import numpy as np import numpy.random as rn import matplotlib.pyplot as plt # to plot import matplotlib as mpl from scipy import optimize # to compare import seaborn as sns sns.set(context="talk", style="darkgrid", palette="hls", font="sans-serif", font_scale=1.05) FIGSIZE = (19, 8) #: Figure size, in inches! mpl.rcParams['figure.figsize'] = FIGSIZE # ---- # # ## Algorithm # # The following pseudocode presents the simulated annealing heuristic. # # - It starts from a state $s_0$ and continues to either a maximum of $k_{\max}$ steps or until a state with an energy of $e_{\min}$ or less is found. # - In the process, the call $\mathrm{neighbour}(s)$ should generate a randomly chosen neighbour of a given state $s$. # - The annealing schedule is defined by the call $\mathrm{temperature}(r)$, which should yield the temperature to use, given the fraction $r$ of the time budget that has been expended so far. # > **Simulated Annealing**: # > # > - Let $s$ = $s_0$ # > - For $k = 0$ through $k_{\max}$ (exclusive): # > + $T := \mathrm{temperature}(k ∕ k_{\max})$ # > + Pick a random neighbour, $s_{\mathrm{new}} := \mathrm{neighbour}(s)$ # > + If $P(E(s), E(s_{\mathrm{new}}), T) \geq \mathrm{random}(0, 1)$: # > * $s := s_{\mathrm{new}}$ # > - Output: the final state $s$ # ---- # # ## Basic but generic Python code # Let us start with a very generic implementation: # In[3]: def annealing(random_start, cost_function, random_neighbour, acceptance, temperature, maxsteps=1000, debug=True): """ Optimize the black-box function 'cost_function' with the simulated annealing algorithm.""" state = random_start() cost = cost_function(state) states, costs = [state], [cost] for step in range(maxsteps): fraction = step / float(maxsteps) T = temperature(fraction) new_state = random_neighbour(state, fraction) new_cost = cost_function(new_state) if debug: print("Step #{:>2}/{:>2} : T = {:>4.3g}, state = {:>4.3g}, cost = {:>4.3g}, new_state = {:>4.3g}, new_cost = {:>4.3g} ...".format(step, maxsteps, T, state, cost, new_state, new_cost)) if acceptance_probability(cost, new_cost, T) > rn.random(): state, cost = new_state, new_cost states.append(state) costs.append(cost) # print(" ==> Accept it!") # else: # print(" ==> Reject it...") return state, cost_function(state), states, costs # ---- # # ## Basic example # # We will use this to find the global minimum of the function $x \mapsto x^2$ on $[-10, 10]$. # In[4]: interval = (-10, 10) def f(x): """ Function to minimize.""" return x ** 2 def clip(x): """ Force x to be in the interval.""" a, b = interval return max(min(x, b), a) # In[5]: def random_start(): """ Random point in the interval.""" a, b = interval return a + (b - a) * rn.random_sample() # In[6]: def cost_function(x): """ Cost of x = f(x).""" return f(x) # In[7]: def random_neighbour(x, fraction=1): """Move a little bit x, from the left or the right.""" amplitude = (max(interval) - min(interval)) * fraction / 10 delta = (-amplitude/2.) + amplitude * rn.random_sample() return clip(x + delta) # In[8]: def acceptance_probability(cost, new_cost, temperature): if new_cost < cost: # print(" - Acceptance probabilty = 1 as new_cost = {} < cost = {}...".format(new_cost, cost)) return 1 else: p = np.exp(- (new_cost - cost) / temperature) # print(" - Acceptance probabilty = {:.3g}...".format(p)) return p # In[9]: def temperature(fraction): """ Example of temperature dicreasing as the process goes on.""" return max(0.01, min(1, 1 - fraction)) # Let's try! # In[10]: annealing(random_start, cost_function, random_neighbour, acceptance_probability, temperature, maxsteps=30, debug=True); # Now with more steps: # In[11]: state, c, states, costs = annealing(random_start, cost_function, random_neighbour, acceptance_probability, temperature, maxsteps=1000, debug=False) state c # ---- # # ## Visualizing the steps # In[12]: def see_annealing(states, costs): plt.figure() plt.suptitle("Evolution of states and costs of the simulated annealing") plt.subplot(121) plt.plot(states, 'r') plt.title("States") plt.subplot(122) plt.plot(costs, 'b') plt.title("Costs") plt.show() # In[13]: see_annealing(states, costs) # ---- # # ## More visualizations # In[14]: def visualize_annealing(cost_function): state, c, states, costs = annealing(random_start, cost_function, random_neighbour, acceptance_probability, temperature, maxsteps=1000, debug=False) see_annealing(states, costs) return state, c # In[15]: visualize_annealing(lambda x: x**3) # In[16]: visualize_annealing(lambda x: x**2) # In[17]: visualize_annealing(np.abs) # In[18]: visualize_annealing(np.cos) # In[19]: visualize_annealing(lambda x: np.sin(x) + np.cos(x)) # In all these examples, the simulated annealing converges to a global minimum. # It can be non-unique, but it is found. # ---- # > That's it for today, folks! # # More notebooks can be found on [my GitHub page](https://GitHub.com/Naereen/notebooks).