-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdeep-water-creatures.c
209 lines (164 loc) · 6.75 KB
/
deep-water-creatures.c
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
// Deep Water Creatures Experiment using Genetic Algorithm
#include <stdio.h>
#include <time.h>
#include <stdlib.h>
#define max_pairs 1000
#define bug_offsets 40
int max_gen = 4004; // max number of generations
int max_energy = 0; // initial max energy
int feed_days = 100; // feed days for the creatures
int prob = 400; // 1/prob is propbability to apply mutation
int en[max_pairs]; // energy array for all pairs
int oq[max_pairs][bug_offsets]; // 1000 e-bug pairs;
// creature 1: from oq[][0] - oq[][19]
// - response to the environment: oq[][0]- oq[][9];
// - deployment of food tenticles: oq[][10]-oq[][19];
// creature 2: from oq[][20] - oq[][39]
// - response to the environment: oq[][20]- oq[][29];
// - deployment of food tenticles: oq[][30]-oq[][39];
int f1, f2; // food items for each creature
int total_en_this_gen;
int gen, tmp, tmp1, tmp2;
FILE *f_out;
// ------------ MAIN ------------
int main () {
int c, i,j, g, k, l, n, s1, s2, r, r1, r2, t;
int next_gen(), save_state();
// initialize pairs at random;
for (i = 0; i < max_pairs; i++)
for (j = 0; j < bug_offsets; j++) oq[i][j] = rand() % 10;
// ---------- main generation loop begins ---------------
for (gen = 0; gen < max_gen; gen++) {
printf("Generation %d\n",gen);
// init energy for each eBug duo
for (i = 0; i < max_pairs; i++)
en[i] = 0;
// feed the creatures in a single generation before the creation of the next generation
for (c = 0; c < feed_days; c++) {
// generate the number of food items:
// - f1 above eBug1
// - f2 above eBug2
f1 = rand() % 10;
f2 = rand() % 10;
// subject every pair of eBugs to the generated environment
for (i = 0; i < max_pairs; i++) {
s2 = oq[i][f2]; // the 1st bug sees f2 items above bug2 and returns signal s2
s1 = oq[i][f1+20]; // the 2nd bug sees f1 items above bug1 and returns signal s1
// generate responses
r1 = oq[i][10+s1]; // r1 = the number of food tenticles deployed by bug1
r2 = oq[i][30+s2]; // r2 = the number of food tenticles deployed by bug2
// adjust energy after capture of food;
// hit: a catch adds 1 energy unit
// miss: a miss subtracts 1 energy unit
if (r1 < f1) en[i] = en[i] + r1;
else {
en[i] = en[i] + f1; // increase energy for every food found
en[i] = en[i] - (r1 - f1); // pay for the miss
}
if (r2 < f2) en[i] = en[i] + r2;
else {
en[i] = en[i] + f2; // increase energy for every food found
en[i]=en[i] - (r2 - f2); // pay for the miss
}
}
}
// call the next generation if the max has not been reached
if (gen < (max_gen - 1)) next_gen();
}
// save the last generation
save_state();
}
// ------------ NEXT GENERATION ------------
int next_gen () {
int a, c1, i,j,l,d,e,m,n,x1,y1,s,r,v,e_max, e_min, e_min_p, t, z;
int pairs[max_pairs][bug_offsets];
int roulette[max_pairs];
float tp;
// get max energy, e_max, e_min
e_max = 0; e_min = 3000000; e_min_p = 3000000;
total_en_this_gen = 0;
c1 = 0;
// adjust various counters
for (i = 0; i < max_pairs; i++) {
if (en[i] < e_min) e_min = en[i];
if (en[i] > e_max) e_max = en[i];
if ((en[i] < e_min_p) && (en[i]>0)) e_min_p = en[i];
if (en[i] > max_energy) {
max_energy = en[i];
tmp1 = gen;
tmp2 = i;
}
if (en[i] > 0) c1++; // c1=number of pairs alive
}
printf("MIN energy this gen = %d\nMIN_pos energy this gen = %d\nMAX energy this gen = %d\nPairs Alive = %d\n",
e_min, e_min_p, e_max, c1);
printf("Max en so far = %d in gen = %d with e-bug = %d\n",max_energy, tmp1, tmp2);
// assign roulette values to each heave and store them in roulette[]
// estimate portion of roulette assigned to a heave:
// - assign 100 - offset slots minus "offset" to max and, in general,
// assign (e[i]/e_max)*100 - (e_min/e_max)*100
// =((e[i] - e_min)/e_max)*100
r = 0;
for (i = 0; i < max_pairs; i++) {
if ((en[i] < 1) || (en[i] == e_min_p)) a = 1; // if energy < 0 assign just 1 slot
else {
tp = ((en[i] - e_min_p)/(1.0 * e_max))*100;
a = tp;
if ((tp - a) > 0.5 ) a++;
}
r = r + a;
roulette[i] = r;
}
printf("Size of roulette = %d \n------------------------\n",r);
// crossover; start populating pairs[][]; there are max_pairs/2 pairs to be formed
for (i = 0; i < (max_pairs/2); i++) {
// find parents with probability prop. to number of spots in the roulette
x1 = rand() % r; // slot value x1 selected
/*************** NEED EXPLANATION ***************/
// find parent which has x1 in its roulette slots
for (j = 0; j < max_pairs; j++)
// make x1 the id of the first parent
if (x1 < roulette[j]) {
x1 = j;
j = 600;
}vi d
// look at Fig. 1. If x1=60, then red[3] (or x1=3) is selected as 60 belongs to roulette slot of red[3]
// find the second parent -- y1
do
{ y1=rand() % r;
for (j=0; j < 500; j++)
if (y1 < roulette[j]) {y1=j; j=600;}
}
while (x1 == y1); // make sure that parents are different
d=2*i; e=(2*i) + 1; // d, e -- children
// make cross-over with probability 0.7: O's with O's and Q's with Q's
if ((rand() % 10) < 7) // if rand <7 do cross over
{
for (j=0; j<40; j++) // copy action vectors switching at random
{ if ((rand() % 2) == 0) // if 0, x1-->d, y1-->e
{pairs[d][j]=oq[x1][j]; pairs[e][j]=oq[y1][j];}
else {pairs[d][j]=oq[y1][j]; pairs[e][j]=oq[x1][j];}
}
// randomly mutate z'th bit with small probability 1/prob
for (z=0; z<40; z++)
{if ((rand() % prob) == 20) pairs[d][z] = rand() % 10;
if ((rand() % prob) == 20) pairs[e][z] = rand() % 10;
}
}
else // no crossover -- just copy
for (z=0; z<40; z++)
{pairs[d][z]=oq[x1][z]; pairs[e][z]=oq[y1][z];}
}
// copy pairs to heave collection
for (i=0; i<500; i++)
for (j=0; j<40; j++)
oq[i][j]=pairs[i][j];
}
/*************** NEED EXPLANATION ***************/
// ------------------ SAVE STATE ------------------
int save_state() {
int i,j;
f_out = fopen("data_out","w");
for (i = 0; i < max_pairs; i++)
for (j = 0; j < bug_offsets; j++) fprintf(f_out,"%d ",oq[i][j]);
}