-
Notifications
You must be signed in to change notification settings - Fork 3
/
simulated_annealing.py
233 lines (183 loc) · 11.9 KB
/
simulated_annealing.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
#!python3
# Copyright (C) 2020 Victor O. Costa
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# Python standard lib
import math
# 3rd party
import numpy as np
# Own
from base_metaheuristic import Base
class SA(Base):
""" Class for the Simulated Annealing optimizer (Kirkpatrick et al., 1983) with perturbation on continuous variable as in (Geng and Marmarelis, 2016) and using exponential decay cooling schedule (Nourani and Andresen, 1998) """
def __init__(self):
""" Constructor """
# Define verbosity and NULL problem definition
super().__init__
# Initial algorithm parameters
self.relative_iterations = None # Array containing the iterations at which best solutions are reported
self.num_local_iter = 0 # Maximum number of local iterations
self.temperature = 100.0 # Temperature, which determines acceptance probability of Metropolis sampling
self.cooling_constant = 0.99 # Parameter for the exponential decay cooling schedule
self.step_size = 1e-2 # Technically each variable type has its own step size, but (Geng and Marmarelis, 2016) uses the same step for all variables
# Optimization results
self.chosen_variable = None
self.current_solution = None # Set of variables that define the current solution, with its cost as the last element of the list
self.best_solution = None # Best solution of the archive
def set_parameters(self, initial_temperature, cooling_constant, step_size, num_local_iter, function_evaluations_array):
""" Define values for the parameters used by the algorithm """
# Input error checking
if num_local_iter <= 0 or initial_temperature <= 0 or cooling_constant <= 0 or step_size <= 0:
print("Error, parameters must be non-null positives")
exit(-1)
if len(function_evaluations_array) == 0:
print("Error, objective function evaluation array must not be empty")
exit(-1)
# Number of function evaluations for SA: global_iterations * local_iterations
# global_iterations = np.max(function_evals_array) / local_iterations
function_evaluations_array = np.array(function_evaluations_array)
self.relative_iterations = (function_evaluations_array) / num_local_iter
all_divisible = (np.array([x.is_integer() for x in self.relative_iterations])).all()
if not all_divisible:
print("Error, at least one number of function evaluations is not divisible by the number of local iterations")
exit(-1)
self.num_global_iter = int(np.max(self.relative_iterations))
self.num_local_iter = num_local_iter
self.temperature = initial_temperature
self.cooling_constant = cooling_constant
self.step_size = step_size
def define_variables(self, initial_ranges, is_bounded):
""" Defines the number of variables, their initial values ranges and wether or not these ranges constrain the variable during the search """
# Input error checking
if self.num_global_iter == 0 or self.num_local_iter == 0:
print("Error, please set algorithm parameters before variables definition")
exit(-1)
if len(initial_ranges) == 0 or len(is_bounded) == 0:
print("Error, initial_ranges and is_bounded lists must not be empty")
exit(-1)
if len(initial_ranges) != len(is_bounded):
print("Error, the number of variables for initial_ranges and is_bounded must be equal")
exit(-1)
self.num_variables = len(initial_ranges)
self.initial_ranges = initial_ranges
self.is_bounded = is_bounded
self.current_solution = np.zeros(self.num_variables + 1)
def compute_perturbation(self):
""" For vanilla SA, the perturbation has fixed size for all variables """
# Random sign of perturbation
random_sign = (-1) ** np.random.randint(0,2)
perturbation = random_sign * self.step_size
return perturbation
# Feedback functions over Bates distribution standard deviation in ACFSA
# No effect on vanilla SA
def positive_feedback(self):
pass
def negative_feedback(self):
pass
def optimize(self):
""" Generate a random initial solution and enter the algorithm loop until the number of global iterations is reached """
# Input error checking
if self.num_variables == None:
print("Error, first set the number of variables and their boundaries")
exit(-1)
if self.cost_function == None:
print("Error, first define the cost function to be used")
exit(-1)
# Randomize initial solution
for i in range(self.num_variables):
self.current_solution[i] = np.random.uniform(self.initial_ranges[i][0], self.initial_ranges[i][1])
# Compute its cost considering that weights were modified
self.current_solution[-1] = self.cost_function(self.current_solution[:-1])
self.best_solution = np.array(self.current_solution)
# Keep solutions defined by function_evaluations_array
recorded_solutions = []
if self.verbosity: print("[ALGORITHM MAIN LOOP]")
# SA main loop
for global_i in range(self.num_global_iter):
# Update temperature according to the exponential decay cooling scheduling
self.temperature = self.temperature * self.cooling_constant
for local_i in range(self.num_local_iter):
total_i = local_i + self.num_local_iter * global_i
if self.verbosity:
print("[%d]" % total_i)
print(self.current_solution)
## Generate candidate solution and compute its cost
# Choose which variable will be pertubated
self.chosen_variable = np.random.randint(0, self.num_variables) # [0, num_variables)
# Perturbate the chosen variable
perturbation = self.compute_perturbation()
pertubated_variable = self.current_solution[self.chosen_variable] + perturbation
# For bounded variables, deal with search space violation using the hard border strategy
if self.is_bounded[self.chosen_variable]:
if pertubated_variable < self.initial_ranges[self.chosen_variable][0]:
pertubated_variable = self.initial_ranges[self.chosen_variable][0]
elif pertubated_variable > self.initial_ranges[self.chosen_variable][1]:
pertubated_variable = self.initial_ranges[self.chosen_variable][1]
candidate_solution = np.array(self.current_solution)
candidate_solution[self.chosen_variable] = pertubated_variable
candidate_solution[-1] = self.cost_function(candidate_solution[:-1])
# Decide if solution will replace the current one based on the Metropolis sampling algorithm
delta_J = candidate_solution[-1] - self.current_solution[-1]
if delta_J < 0:
acceptance_probability = 1.0
# Possibly update best solution seen during search until the moment
if candidate_solution[-1] < self.best_solution[-1]:
self.best_solution = np.array(candidate_solution)
else:
acceptance_probability = math.exp(-delta_J/self.temperature)
# Candidate accepted
if np.random.rand() <= acceptance_probability:
self.current_solution[self.chosen_variable] = candidate_solution[self.chosen_variable]
self.current_solution[-1] = candidate_solution[-1]
# Positive feedback over Bates distribution standard deviation in ACFSA
# Has no effect in vanilla SA
self.positive_feedback()
# Candidate rejected
else:
# Negative feedback over Bates distribution standard deviation in ACFSA
# Has no effect in vanilla SA
self.negative_feedback()
if (self.relative_iterations - 1 == global_i).any():
recorded_solutions.append(np.array(self.best_solution))
return np.array(recorded_solutions)
class ACFSA(SA):
""" Simulated annealing using adaptive solution generation based in the feedback (positive feedback C and negative feedback) heuristics described in (Martins et al., 2012) """
def __init__(self):
""" Constructor """
# Define verbosity and NULL problem definition
super().__init__
self.crystallization_factor = None # crystallization factors define the starndard deviation of the step size distribution for each variable at each itertion
def set_parameters(self, initial_temperature, cooling_constant, num_local_iter, function_evaluations_array):
""" Define values for the parameters used by the algorithm """
super().set_parameters(initial_temperature, cooling_constant, 1, num_local_iter, function_evaluations_array)
def define_variables(self, initial_ranges, is_bounded):
""" Defines the number of variables, their initial values ranges and wether or not these ranges constrain the variable during the search.
Defines crystallization_factor dimensionality """
super().define_variables(initial_ranges, is_bounded)
self.crystallization_factor = np.ones(self.num_variables)
def compute_perturbation(self):
""" In ACFSA, perturbation is generated by sampling from Bates distribution,
with standard deviation inversely proportional to the square root of the crystallization_factor for the given variable """
if not (self.crystallization_factor[self.chosen_variable] % 1 == 0.0):
print("Crystallization factor must be an integer")
exit(-1)
random_array = np.random.uniform(low = -0.5, high = 0.5, size = int(self.crystallization_factor[self.chosen_variable]))
perturbation = np.sum(random_array) / self.crystallization_factor[self.chosen_variable]
return perturbation
def positive_feedback(self):
""" Increases standard deviation of Bates distribution in perturbation """
self.crystallization_factor[self.chosen_variable] = math.ceil(self.crystallization_factor[self.chosen_variable] / 4)
if self.crystallization_factor[self.chosen_variable] == 0:
self.crystallization_factor[self.chosen_variable] = 1
def negative_feedback(self):
""" Decreases standard deviation of Bates distribution in perturbation """
if self.crystallization_factor[self.chosen_variable] < int(1e5):
self.crystallization_factor[self.chosen_variable] += 1