genetically_evolving_neural_network
This package mimicks the process of Genetic Evolution on Entities, each comprised of a Neural Network, through cross-breeding parents and genetic mutations.
Table of Contents
Usage
You simply need to define your own FitnessService
to determine how well an Entity's decisions perform, along with a config that states how often the DNA of your population should mutate.
How it works
This library will generate a randomized initial population of Neural Network Entities. You can think of an Entity's DNA as being a Neural Network, and the genes within that DNA being perceptrons within the Neural Network.
Each Neural Network is evaluated against a FitnessService
and assigned a fitness score based on how well it performs a task.
Entities with a higher fitness score have a higher probability of being chosen as a parent to reproduce and create offspring for the next generation.
Genes from each parent are taken to create the child's new DNA, and there is a chance that those genes will mutate (just like how reproduction and evolution happen in real life).
This cycle repeats itself, keeping the high performing Neural Networks within the gene pool and allowing for the process of evolution through mutations.
To Begin:
- Define your own
FitnessService
that assigns a fitness score to an entity based on how well it performs. This could mean guessing if a number is positive or negative, choosing a next move in a chess game, or any other decision a neural network can make. - Define your
GeneticEvolutionConfig
, responsible for how many entities should exist within a population, how often an Entity's DNA should mutate, and other adjustable values.
Positive or Negative Number Classifier Example
In this example, we will guess whether an input is positive or negative.
(You can view the entire file here)
First, define the GENNFitnessService
that rewards correctly identifying if a number is positive or negative.
/// The inputs for this Neural Network, from -1 to 1 in increments of 0.1.
List<double> get inputs => List.generate(10, (index) => index * 0.1)
..addAll(List.generate(9, (index) => (index + 1) * -0.1));
/// The scoring function that will be used to evolve entities of a population
class PositiveNumberFitnessService extends GENNFitnessService {
@override
Future<double> gennScoringFunction({
required GENNNeuralNetwork neuralNetwork,
}) async {
// Calculate how many correct guesses were made
return inputs.fold(0, (previousValue, input) {
final guess = neuralNetwork.guess(inputs: [input])[0];
// Only add a point if the neural network guesses correctly
if ((input > 0 && guess > 0) || (input <= 0 && guess == 0)) {
return previousValue + 1;
}
return previousValue;
}).toDouble();
}
}
Next, you initialize a GeneticEvolution
object, responsible for creating populations of child Entities.
final config = GENNGeneticEvolutionConfig(
numInitialInputs: 1, // 1 input into neural network
numOutputs: 1, // 1 output from neural network
layerMutationRate: 0.1, // 10% chance to add/remove layer to network
perceptronMutationRate: 0.2, // 20% chance to add/remove perceptron to layer
mutationRate: 0.05, // 5% chance to mutate an existing perceptron
);
Next, you can use the fitness service and config to create a GENN
object.
final genn = GENN.create(
config: config,
fitnessService: PositiveNumberFitnessService(),
);
And finally, you can access the next generation of Entities using nextGeneration()
.
final nextGen = await genn.nextGeneration();
File Parsing
You can write specific generations to a file, and similarly read in specific generations stored on a file.
In order to write the current GENNGeneration
object to a file
genn.writeGenerationToFile();
In order to load in a specific GENNGeneration
read from a file
genn.loadGenerationFromFile(wave: 10);
More Complex Examples
Guessing if a number is positive or negative is not a particularly impressive use case for a Neural Network, but it is very simple to understand and helps grasp the concept. If you'd like to see a more complex problem, check out these examples.