-
Notifications
You must be signed in to change notification settings - Fork 0
/
Species.pde
209 lines (180 loc) · 7.93 KB
/
Species.pde
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
class Species {
ArrayList<Player> players = new ArrayList<Player>();
float bestFitness = 0;
Player champ;
float averageFitness = 0;
int staleness = 0;//how many generations the species has gone without an improvement
Genome rep;
//--------------------------------------------
//coefficients for testing compatibility
float excessCoeff = 1;
float weightDiffCoeff = 0.5;
float compatibilityThreshold = 3;
//------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//empty constructor
Species() {
}
//------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//constructor which takes in the player which belongs to the species
Species(Player p) {
players.add(p);
//since it is the only one in the species it is by default the best
bestFitness = p.fitness;
rep = p.brain.clone();
champ = p.cloneForReplay();
}
//------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//returns whether the parameter genome is in this species
boolean sameSpecies(Genome g) {
float compatibility;
float excessAndDisjoint = getExcessDisjoint(g, rep);//get the number of excess and disjoint genes between this player and the current species rep
float averageWeightDiff = averageWeightDiff(g, rep);//get the average weight difference between matching genes
float largeGenomeNormaliser = g.genes.size() - 20;
if (largeGenomeNormaliser<1) {
largeGenomeNormaliser =1;
}
compatibility = (excessCoeff* excessAndDisjoint/largeGenomeNormaliser) + (weightDiffCoeff* averageWeightDiff);//compatablilty formula
return (compatibilityThreshold > compatibility);
}
//------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//add a player to the species
void addToSpecies(Player p) {
players.add(p);
}
//------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//returns the number of excess and disjoint genes between the 2 input genomes
//i.e. returns the number of genes which dont match
float getExcessDisjoint(Genome brain1, Genome brain2) {
float matching = 0.0;
for (int i =0; i <brain1.genes.size(); i++) {
for (int j = 0; j < brain2.genes.size(); j++) {
if (brain1.genes.get(i).innovationNo == brain2.genes.get(j).innovationNo) {
matching ++;
break;
}
}
}
return (brain1.genes.size() + brain2.genes.size() - 2*(matching));//return no of excess and disjoint genes
}
//------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//returns the avereage weight difference between matching genes in the input genomes
float averageWeightDiff(Genome brain1, Genome brain2) {
if (brain1.genes.size() == 0 || brain2.genes.size() ==0) {
return 0;
}
float matching = 0;
float totalDiff= 0;
for (int i =0; i <brain1.genes.size(); i++) {
for (int j = 0; j < brain2.genes.size(); j++) {
if (brain1.genes.get(i).innovationNo == brain2.genes.get(j).innovationNo) {
matching ++;
totalDiff += abs(brain1.genes.get(i).weight - brain2.genes.get(j).weight);
break;
}
}
}
if (matching ==0) {//divide by 0 error
return 100;
}
return totalDiff/matching;
}
//-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//sorts the species by fitness
void sortSpecies() {
ArrayList<Player> temp = new ArrayList<Player>();
//selection short
for (int i = 0; i < players.size(); i ++) {
float max = 0;
int maxIndex = 0;
for (int j = 0; j< players.size(); j++) {
if (players.get(j).fitness > max) {
max = players.get(j).fitness;
maxIndex = j;
}
}
temp.add(players.get(maxIndex));
players.remove(maxIndex);
i--;
}
players = (ArrayList)temp.clone();
if (players.size() == 0) {
print("fucking");
staleness = 200;
return;
}
//if new best player
if (players.get(0).fitness > bestFitness) {
staleness = 0;
bestFitness = players.get(0).fitness;
rep = players.get(0).brain.clone();
champ = players.get(0).cloneForReplay();
//storechamp();
} else {//if no new best player
staleness ++;
}
}
//-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//simple stuff
void setAverage() {
float sum = 0;
for (int i = 0; i < players.size(); i ++) {
sum += players.get(i).fitness;
}
averageFitness = sum/players.size();
}
//-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//gets baby from the players in this species
Player giveMeBaby(ArrayList<connectionHistory> innovationHistory) {
Player baby;
if (random(1) < 0.25) {//25% of the time there is no crossover and the child is simply a clone of a random(ish) player
baby = selectPlayer().clone();
} else {//75% of the time do crossover
//get 2 random(ish) parents
Player parent1 = selectPlayer();
Player parent2 = selectPlayer();
//the crossover function expects the highest fitness parent to be the object and the lowest as the argument
if (parent1.fitness < parent2.fitness) {
baby = parent2.crossover(parent1);
} else {
baby = parent1.crossover(parent2);
}
}
baby.brain.mutate(innovationHistory);//mutate that baby brain
return baby;
}
//-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
//selects a player based on it fitness
Player selectPlayer() {
float fitnessSum = 0;
for (int i =0; i<players.size(); i++) {
fitnessSum += players.get(i).fitness;
}
float rand = random(fitnessSum);
float runningSum = 0;
for (int i = 0; i<players.size(); i++) {
runningSum += players.get(i).fitness;
if (runningSum > rand) {
return players.get(i);
}
}
//unreachable code to make the parser happy
return players.get(0);
}
//------------------------------------------------------------------------------------------------------------------------------------------
//kills off bottom half of the species
void cull() {
if (players.size() > 2) {
for (int i = players.size()/2; i<players.size(); i++) {
players.remove(i);
i--;
}
}
}
//------------------------------------------------------------------------------------------------------------------------------------------
//in order to protect unique players, the fitnesses of each player is divided by the number of players in the species that that player belongs to
void fitnessSharing() {
for (int i = 0; i< players.size(); i++) {
players.get(i).fitness/=players.size();
}
}
}