20 optional<Node> newNode =
SS.get_node_like(spot.node->data);
26 Tree.replace(spot, *newNode);
44 vector<float> weights;
47 Iter iter = Tree.begin();
48 std::transform(Tree.begin(), Tree.end(), std::back_inserter(weights),
50 size_t d = 1+Tree.depth(iter);
51 std::advance(iter, 1);
54 if ((d >= params.get_max_depth())
55 || (SS.node_map.find(n.ret_type) == SS.node_map.end())) {
59 return n.get_prob_change();
65 weights.resize(Tree.size());
66 std::fill(weights.begin(), weights.end(), 0.0f);
75 auto spot_type = spot.node->data.ret_type;
84 std::optional<Node> n =
SS.sample_op_with_arg(
85 spot_type, spot_type,
true, params.
max_size-Tree.size()-1);
91 auto parent_node = Tree.wrap(spot, *n);
94 bool spot_filled =
false;
95 for (
auto a: (*n).arg_types)
100 auto opt =
SS.sample_terminal(a);
105 Tree.append_child(parent_node, opt.value());
108 else if (a == spot_type)
112 auto opt =
SS.sample_terminal(a);
117 Tree.insert(spot, opt.value());
139 auto opt =
SS.sample_terminal(spot.node->data.ret_type);
144 Tree.erase_children(spot);
146 Tree.replace(spot, opt.value());
164 vector<float> weights(Tree.size());
166 if (Tree.size() < params.
max_size) {
167 std::transform(Tree.begin(), Tree.end(), weights.begin(),
170 if (Is<NodeType::OffsetSum>(n.node_type))
174 if (!n.get_is_weighted()
175 && IsWeighable(n.ret_type))
177 return n.get_prob_change();
185 std::fill(weights.begin(), weights.end(), 0.0f);
194 if (spot.node->data.get_is_weighted()==
true
198 spot.node->data.set_is_weighted(
true);
215 vector<float> weights(Tree.size());
217 std::transform(Tree.begin(), Tree.end(), weights.begin(),
220 if (Is<NodeType::OffsetSum>(n.node_type))
223 if (n.get_is_weighted()
224 && IsWeighable(n.ret_type))
225 return n.get_prob_change();
238 if (spot.node->data.get_is_weighted()==
false)
241 spot.node->data.set_is_weighted(
false);
258 vector<float> weights;
260 auto node_map =
SS.node_map;
262 if (Tree.size() < params.
max_size) {
263 Iter iter = Tree.begin();
264 std::transform(Tree.begin(), Tree.end(), std::back_inserter(weights),
266 size_t d = 1+Tree.depth(iter);
267 std::advance(iter, 1);
270 if ((d >= params.max_depth)
271 || (SS.node_map.find(n.ret_type) == SS.node_map.end())
272 || (SS.node_map.find(n.ret_type) == SS.node_map.end()) )
275 return n.get_prob_change();
279 weights.resize(Tree.size());
280 std::fill(weights.begin(), weights.end(), 0.0f);
293 if ( params.
max_size <= (Tree.size() - Tree.size(spot))
294 || params.
max_depth <= Tree.depth(spot) )
297 auto spot_type = spot.node->data.ret_type;
301 size_t d = params.
max_depth - Tree.depth(spot);
302 size_t s = params.
max_size - (Tree.size() - Tree.size(spot));
308 auto subtree =
SS.sample_subtree(spot.node->data, d, s);
314 Tree.erase_children(spot);
315 Tree.replace(spot, subtree.value().begin());
344template<Brush::ProgramType T>
354 vector<float> child_weights(child.
Tree.size());
355 auto child_iter = child.
Tree.begin();
356 std::transform(child.
Tree.begin(), child.
Tree.end(), child_weights.begin(),
358 auto s_at = child.size_at(child_iter);
359 auto d_at = child.depth_to_reach(child_iter);
361 std::advance(child_iter, 1);
363 if (s_at<parameters.max_size && d_at<parameters.max_depth)
364 return n.get_prob_change();
370 if (std::all_of(child_weights.begin(), child_weights.end(), [](
const auto& w) {
381 while (++attempts <= 3)
383 auto child_spot =
r.select_randomly(child.
Tree.begin(),
385 child_weights.begin(),
389 auto child_ret_type = child_spot.node->data.ret_type;
396 vector<float> other_weights(other.
Tree.size());
399 auto other_iter = other.
Tree.begin();
402 const auto check_and_incrm = [other, &other_iter, allowed_size, allowed_depth]() ->
bool {
403 int s = other.
size_at( other_iter );
404 int d = other.
depth_at( other_iter );
406 std::advance(other_iter, 1);
407 return (s <= allowed_size) && (d <= allowed_depth);
410 std::transform(other.
Tree.begin(), other.
Tree.end(),
411 other_weights.begin(),
412 [child_ret_type, check_and_incrm](
const auto& n){
415 if (check_and_incrm() && (n.ret_type == child_ret_type))
416 return n.get_prob_change();
423 bool matching_spots_found =
false;
424 for (
const auto& w: other_weights)
427 matching_spots_found = w > 0.0;
429 if (matching_spots_found) {
430 auto other_spot =
r.select_randomly(
433 other_weights.begin(),
439 child.
Tree.move_ontop(child_spot, other_spot);
485template<Brush::ProgramType T>
490 bool all_zero =
true;
492 if (it.second > 0.0) {
506 while(++attempts <= 3)
509 string choice =
r.random_choice(
parameters.mutation_probs);
511 vector<float> weights;
514 if (choice ==
"point")
516 else if (choice ==
"insert")
518 else if (choice ==
"delete")
520 else if (choice ==
"subtree")
522 else if (choice ==
"toggle_weight_on")
524 else if (choice ==
"toggle_weight_off")
527 std::string msg = fmt::format(
"{} not a valid mutation choice", choice);
531 if (std::all_of(weights.begin(), weights.end(), [](
const auto& w) {
539 auto spot =
r.select_randomly(child.
Tree.begin(), child.
Tree.end(),
540 weights.begin(), weights.end());
547 if (choice ==
"point")
549 else if (choice ==
"insert")
551 else if (choice ==
"delete")
553 else if (choice ==
"subtree")
555 else if (choice ==
"toggle_weight_on")
577template <Brush::ProgramType T>
579 const vector<size_t>& parents)
583 for (
unsigned i = 0; i<indices.size(); ++i)
591 std::optional<Individual<T>> opt=std::nullopt;
594 *
r.select_randomly(parents.begin(), parents.end())];
596 vector<Individual<T>> ind_parents;
600 *
r.select_randomly(parents.begin(), parents.end())];
602 opt =
cross(mom, dad);
603 ind_parents = {mom, dad};
624 pop.
individuals.at(indices.at(i)) = std::make_shared<Individual<T>>(ind);
635 pop.
individuals.at(indices.at(i)) = std::make_shared<Individual<T>>(new_ind);
vector< string > get_objectives() const
void set_objectives(vector< string > objs)
void set_parents(const vector< Individual< T > > &parents)
void init(SearchSpace &ss, const Parameters ¶ms)
Program< T > program
executable data structure
vector< size_t > get_island_indexes(int island)
vector< std::shared_ptr< Individual< T > > > individuals
delete subtree and replace it with a terminal of the same return type
static auto mutate(tree< Node > &Tree, Iter spot, const SearchSpace &SS, const Parameters ¶ms)
insert a node with spot as a child
static auto find_spots(tree< Node > &Tree, const SearchSpace &SS, const Parameters ¶ms)
static auto mutate(tree< Node > &Tree, Iter spot, const SearchSpace &SS, const Parameters ¶ms)
tree< Node >::pre_order_iterator Iter
static auto find_spots(tree< Node > &Tree, const SearchSpace &SS, const Parameters ¶ms)
replace node with same typed node
static auto mutate(tree< Node > &Tree, Iter spot, const SearchSpace &SS, const Parameters ¶ms)
replaces the subtree rooted in spot
static auto find_spots(tree< Node > &Tree, const SearchSpace &SS, const Parameters ¶ms)
static auto mutate(tree< Node > &Tree, Iter spot, const SearchSpace &SS, const Parameters ¶ms)
toggle the node's weight OFF
static auto find_spots(tree< Node > &Tree, const SearchSpace &SS, const Parameters ¶ms)
static auto mutate(tree< Node > &Tree, Iter spot, const SearchSpace &SS, const Parameters ¶ms)
toggle the node's weight ON
static auto mutate(tree< Node > &Tree, Iter spot, const SearchSpace &SS, const Parameters ¶ms)
static auto find_spots(tree< Node > &Tree, const SearchSpace &SS, const Parameters ¶ms)
std::optional< Individual< T > > mutate(const Individual< T > &parent)
Performs mutation operation on an individual.
void vary(Population< T > &pop, int island, const vector< size_t > &parents)
Handles variation of a population.
std::optional< Individual< T > > cross(const Individual< T > &mom, const Individual< T > &dad)
Performs crossover operation on two individuals.
#define HANDLE_ERROR_THROW(err)
< nsga2 selection operator for getting the front
auto IsWeighable() noexcept -> bool
unsigned get_max_size() const
An individual program, a.k.a. model.
int size_at(Iter &top, bool include_weight=true) const
count the size of a given subtree, optionally including the weights in weighted nodes....
int depth() const
count the tree depth of the program. The depth is not influenced by weighted nodes.
int depth_at(Iter &top) const
count the depth of a given subtree. The depth is not influenced by weighted nodes....
int depth_to_reach(Iter &top) const
count the depth until reaching the given subtree. The depth is not influenced by weighted nodes....
int size(bool include_weight=true) const
count the tree size of the program, including the weights in weighted nodes.
Holds a search space, consisting of operations and terminals and functions, and methods to sample tha...