//! Example of optimizing the Schwefel function with genetic algorithm. //! //! y = f(x), where x = (x0, x1, ..., xi,... xn). //! Global minimum is x' = (420.9687, 420.9687, ...) for any xi lying in [-500.0; 500.0]. //! f(x') = 0 //! //! # Terms //! * `Goal function` - the function for optimization. y = f(x). //! * `Gene` - a single value of xi. //! * `Chromosome` - a point in the search space. x = (x0, x1, x2, ..., xn). //! * `Individual` - union of x and value of goal function. //! * `Population` - set of the individuals. //! * `Generation` - a number of iteration of genetic algorithm. use std::io; use ew::genetic::{self, creation, cross, mutation, pairing, pre_birth, selection}; use ew::tools::logging; use ew::tools::stopchecker; use ew::{GoalFromFunction, IterativeOptimizer, Optimizer}; use ew_testfunc; /// Gene type type Gene = f32; /// Chromosomes type type Chromosomes = Vec; fn main() { // General parameters // Search space. Any xi lies in [-500.0; 500.0] let minval: Gene = -500.0; let maxval: Gene = 500.0; // Count individuals in initial population let population_size = 500; // Count of xi in the chromosomes let chromo_count = 15; let intervals = vec![(minval, maxval); chromo_count]; // Make a trait object for goal function (Schwefel function) let goal = GoalFromFunction::new(ew_testfunc::schwefel); // Make the creator to create initial population. // RandomCreator will fill initial population with individuals with random chromosomes in a // given interval, let creator = creation::vec_float::RandomCreator::new(population_size, intervals.clone()); // Make a trait object for the pairing. // Tournament method. let families_count = population_size / 2; let rounds_count = 5; let pairing = pairing::Tournament::new(families_count).rounds_count(rounds_count); // Crossbreeding algorithm. // Make a Cross trait object. The bitwise crossing for float genes. let single_cross = cross::FloatCrossExp::new(); let cross = cross::VecCrossAllGenes::new(Box::new(single_cross)); // Make a Mutation trait object. // Use bitwise mutation (change random bits with given probability). let mutation_probability = 15.0; let mutation_gene_count = 3; let single_mutation = mutation::BitwiseMutation::new(mutation_gene_count); let mutation = mutation::VecMutation::new(mutation_probability, Box::new(single_mutation)); // Pre birth. Throw away new chlld chromosomes if their genes do not lies if given intervals. let pre_births: Vec>> = vec![Box::new( pre_birth::vec_float::CheckChromoInterval::new(intervals.clone()), )]; // Stop checker. Stop criterion for genetic algorithm. let stop_checker_1 = Box::new(stopchecker::MaxIterations::new(100)); let stop_checker_2 = Box::new(stopchecker::MaxIterations::new(500)); // Make a trait object for selection. Selection is killing the worst individuals. // Kill all individuals if the value of goal function is NaN or Inf. // Kill the worst individuals to population size remained unchanged. let selections: Vec>> = vec![ Box::new(selection::KillFitnessNaN::new()), Box::new(selection::LimitPopulation::new(population_size)), ]; // Make a loggers trait objects // let mut stdout_verbose = io::stdout(); let mut stdout_result = io::stdout(); let mut stdout_time = io::stdout(); let loggers: Vec>> = vec![ // Box::new(logging::VerboseLogger::new(&mut stdout_verbose, 15)), Box::new(logging::ResultOnlyLogger::new(&mut stdout_result, 15)), Box::new(logging::TimeLogger::new(&mut stdout_time)), ]; // Construct main optimizer struct let mut optimizer = genetic::GeneticOptimizer::new( Box::new(goal), stop_checker_1, Box::new(creator), Box::new(pairing), Box::new(cross), Box::new(mutation), selections, pre_births, ); optimizer.set_loggers(loggers); // Run genetic algorithm optimizer.find_min(); optimizer.set_stop_checker(stop_checker_2); optimizer.next_iterations(); }