Skip to content

Commit

Permalink
Fix again constants and rework crossover / selection
Browse files Browse the repository at this point in the history
  • Loading branch information
Vincent Riva committed Mar 30, 2019
1 parent a616ad9 commit a498345
Show file tree
Hide file tree
Showing 3 changed files with 37 additions and 12 deletions.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "genome.js",
"version": "1.1.8",
"version": "1.1.9",
"description": "Genetics algorithms done right",
"main": "dist/genome.min.js",
"scripts": {
Expand Down
15 changes: 15 additions & 0 deletions src/core/Chromosome.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,10 +5,12 @@ export class Chromosome {
private genes: Gene[];
private constants: Gene[];
private fitness: number;
public readonly isKilled: boolean;
constructor(blueprint: Blueprint | null = null) {
this.genes = [];
this.constants = [];
this.fitness = 0;
this.isKilled = false;

if (blueprint) {
this.initializeGenes(blueprint);
Expand Down Expand Up @@ -48,6 +50,19 @@ export class Chromosome {
return chromosome;
}

setDNA(genes: Gene[]) {
this.genes = [];
genes.map((gene: Gene) => {
const geneClone = Chromosome.copyGene(gene);
this.genes.push(geneClone);
});
this.isKilled = false;
}

kill() {
this.isKilled = true;
}

static copyGene(gene: Gene): Gene {
// @ts-ignore
return Object.assign(Object.create(Object.getPrototypeOf(gene)), gene);
Expand Down
32 changes: 21 additions & 11 deletions src/core/Population.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,36 +53,46 @@ export class Population {

selectBestChromosomes() {
const pivot = Math.floor(this.chromosomes.length * this.cutOff);
this.chromosomes.splice(this.chromosomes.length - pivot);
const killedChromosomes = this.chromosomes.slice(this.chromosomes.length - pivot);
const selectedChromosomes = this.chromosomes.slice(0, this.chromosomes.length - pivot + 1);

killedChromosomes.map((chromosome: Chromosome) => {
chromosome.kill();
});

this.sumFitness = 0;
this.chromosomes.map((chromosome: Chromosome) => {
selectedChromosomes.map((chromosome: Chromosome) => {
this.sumFitness += chromosome.getFitness();
});
}

crossoverChromosomes() {
let newChromosomes = [];
for (let i = this.chromosomes.length; i < this.size; i += 1) {
const killedChromosomes = this.chromosomes.filter((chromosome: Chromosome) => {
return chromosome.isKilled;
});

const selectedChromosomes = this.chromosomes.filter((chromosome: Chromosome) => {
return !chromosome.isKilled;
});

for (let i = 0; i < killedChromosomes.length; i += 1) {
let chromosomeA = null;
let chromosomeB = null;

do {
chromosomeA = this.getRandomChromosome();
chromosomeB = this.getRandomChromosome();
chromosomeA = this.getRandomChromosome(selectedChromosomes);
chromosomeB = this.getRandomChromosome(selectedChromosomes);
} while (!chromosomeA || !chromosomeB);

if (chromosomeA && chromosomeB) {
const pivot = Math.floor(Math.random() * chromosomeA.getLength());
const genesA = chromosomeA.getGenes().slice(0, pivot);
const genesB = chromosomeB.getGenes().slice(pivot);
const newChromosome = Chromosome.fromDNA([...genesA, ...genesB], chromosomeA.getConstants());
newChromosomes.push(newChromosome);
killedChromosomes[i].setDNA([...genesA, ...genesB]);
} else {
// console.error('Should not happen');
}
}
this.chromosomes = [...this.chromosomes, ...newChromosomes];
}

setMutationRate(mutationRate: number) {
Expand All @@ -101,10 +111,10 @@ export class Population {
});
}

getRandomChromosome() {
getRandomChromosome(selectedChromosomes: Chromosome[]) {
const random = Math.random() * this.sumFitness;
let sumFitness = 0;
for (const chromosome of this.chromosomes) {
for (const chromosome of selectedChromosomes) {
sumFitness += chromosome.getFitness();
if (random < sumFitness) {
return chromosome;
Expand Down

0 comments on commit a498345

Please sign in to comment.