/* * Encog(tm) Core v3.4 - Java Version * http://www.heatonresearch.com/encog/ * https://github.com/encog/encog-java-core * Copyright 2008-2016 Heaton Research, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * For more information on Heaton Research copyrights, licenses * and trademarks visit: * http://www.heatonresearch.com/copyright */ package org.encog.neural.neat.training.opp; import java.io.Serializable; import java.util.Random; import org.encog.mathutil.randomize.RangeRandomizer; import org.encog.ml.ea.genome.Genome; import org.encog.neural.neat.NEATNeuronType; import org.encog.neural.neat.NEATPopulation; import org.encog.neural.neat.training.NEATGenome; import org.encog.neural.neat.training.NEATNeuronGene; /** * Mutates a NEAT genome by adding a link. To add a link, two random neurons are * chosen and a new random link is created between them. There are severals * rules. Bias and input neurons can never be the target of a link. We also do * not create double links. Two neurons cannot have more than one link going in * the same direction. A neuron can have a single link to itself. * * If the network is only one cycle, then output neurons cannot be a target. * * ----------------------------------------------------------------------------- * http://www.cs.ucf.edu/~kstanley/ Encog's NEAT implementation was drawn from * the following three Journal Articles. For more complete BibTeX sources, see * NEATNetwork.java. * * Evolving Neural Networks Through Augmenting Topologies * * Generating Large-Scale Neural Networks Through Discovering Geometric * Regularities * * Automatic feature selection in neuroevolution */ public class NEATMutateAddLink extends NEATMutation implements Serializable { /** * {@inheritDoc} */ @Override public void performOperation(Random rnd, Genome[] parents, int parentIndex, Genome[] offspring, int offspringIndex) { int countTrysToAddLink = this.getOwner().getMaxTries(); NEATGenome target = this.obtainGenome(parents, parentIndex, offspring, offspringIndex); // the link will be between these two neurons long neuron1ID = -1; long neuron2ID = -1; // try to add a link while ((countTrysToAddLink--) > 0) { final NEATNeuronGene neuron1 = chooseRandomNeuron(target, true); final NEATNeuronGene neuron2 = chooseRandomNeuron(target, false); if (neuron1 == null || neuron2 == null) { return; } // do not duplicate // do not go to a bias neuron // do not go from an output neuron // do not go to an input neuron if (!isDuplicateLink(target, neuron1.getId(), neuron2.getId()) && (neuron2.getNeuronType() != NEATNeuronType.Bias) && (neuron2.getNeuronType() != NEATNeuronType.Input)) { if ( ((NEATPopulation)getOwner().getPopulation()).getActivationCycles() != 1 || neuron1.getNeuronType() != NEATNeuronType.Output) { neuron1ID = neuron1.getId(); neuron2ID = neuron2.getId(); break; } } } // did we fail to find a link if ((neuron1ID < 0) || (neuron2ID < 0)) { return; } double r = ((NEATPopulation) target.getPopulation()).getWeightRange(); createLink(target, neuron1ID, neuron2ID, RangeRandomizer.randomize(rnd, -r, r)); target.sortGenes(); } }