A genetic algorithm implementation for Rust. Inspired by the book Genetic Algorithms in Elixir
There are three main elements to this approach: * The Genotype (the search space) * The Fitness function (the search goal) * The strategy (the search strategy) * Evolve (evolution strategy) * Permutate (for small search spaces, with a 100% guarantee) * HillClimb (when search space is convex with little local optima or when crossover is impossible/inefficient)
Terminology:
* Population: a population has population_size
number of individuals (called chromosomes).
* Chromosome: a chromosome has genes_size
number of genes
* Gene: a gene is a combination of position in the chromosome and value of the gene (allele)
* Allele: alleles are the possible values of the genes
* Genotype: holds the genes_size
and alleles and knows how to generate and mutate chromosomes efficiently
* Fitness: knows how to determine the fitness of a chromosome
See docs.rs
```rust use genetic_algorithm::strategy::evolve::prelude::*;
// the search space let genotype = BinaryGenotype::builder() // boolean alleles .withgenessize(100) // 100 genes per chromosome .build() .unwrap();
println!("{}", genotype);
// the search goal to optimize towards (maximize or minimize)
pub struct CountTrue;
impl Fitness for CountTrue {
type Genotype = BinaryGenotype;
fn calculateforchromosome(&mut self, chromosome: &Chromosome
// the search strategy let mut rng = rand::threadrng(); // a randomness provider implementing Trait rand::Rng let evolve = Evolve::builder() .withgenotype(genotype) .withpopulationsize(100) // evolve with 100 chromosomes .withtargetfitnessscore(100) // goal is 100 times true in the best chromosome .withfitness(CountTrue) // count the number of true values in the chromosomes .withcrossover(CrossoverUniform(true)) // crossover all individual genes between 2 chromosomes for offspring .withmutate(MutateOnce(0.2)) // mutate a single gene with a 20% probability per chromosome .with_compete(CompeteElite) // sort the chromosomes by fitness to determine crossover order .call(&mut rng); .unwrap()
println!("{}", evolve); ```
Run with cargo run --example [EXAMPLE_BASENAME] --release
UniqueGenotype<u8>
with a 64x64 chess board setupNQueensFitness
fitnessBinaryGenotype<(weight, value)>
each gene encodes presence in the knapsackKnapsackFitness(&items, weight_limit)
fitnessDiscreteGenotype<char>
100 monkeys randomly typing characters in a loopRun tests with cargo test
Implemented using criterion. Run benchmarks with cargo bench
Implemented using criterion and pprof.
Find the flamegraph in: ./target/criterion/profile_evolve_binary/profile/flamegraph.svg
Run with cargo run --example profile_evolve_binary --release -- --bench --profile-time 5