Run Several Simulations
To explore a model, the easiest and the most intuitive way to proceed is running several simulations with several parameter value, and see the differences from the output. GAMA provides you the possibility to launch several simulations from the GUI.
Index​
Create a simulation​
Let's remind you that in GAMA, everything is an agent. We already saw that the "world" agent is the agent of the model. The model is thus a species, called modelName_model :
model toto // <- the name of the species is "toto_model"
New highlight of the day : an Experiment is also an agent ! It's a special agent which will instantiate automatically an agent from the model species. You can then perfectly create agents (model agents) from your experiment, using the statement create
:
model multi_simulations // the "world" is an instance of the "multi_simulations_model"
global {
}
experiment my_experiment type:gui {
init {
create multi_simulations_model;
}
}
This sort model will instantiate 2 simulations (two instance of the model) : one is created automatically by the experiment, and the second one is explicitly created through the statement create
.
To simplify the syntax, you can use the built-in attribute simulation
of your experiment. When you have a model called "multi_simulations", the two following lines are strictly equal :
create multi_simulations_model;
create simulation;
As it was the case for creating regular species, you can specify the parameters of your agent during the creation through the facet with:
:
model multi_simulations
global {
rgb bgd_color;
}
experiment my_experiment type:gui {
parameter name:"background color:" var:bgd_color init:#blue;
init {
create simulation with:[bgd_color::#red];
}
output {
display "my_display" background:bgd_color{}
}
}
Manipulate simulations​
When you think the simulations as agents, it gives you a lot of new possibilities. You can for example create a reflex from your experiment, asking to create simulations during the experiment execution !
The following short model for example will create a new simulation at each 10 cycles :
model multi_simulations
global {
init {
write "new simulation created ! Its name is "+name;
}
}
experiment my_experiment type:gui {
init {
}
reflex when:(mod(cycle,10)=0 and cycle!=0) {
create simulation;
}
output {
}
}
You may ask, what is the purpose of such a thing ? Well, with such a short model, it is not very interesting, for sure. But you can imagine running a simulation, and if the simulation reaches a certain state, it can be closed, and another simulation can be run instead with different parameters (a simulation can be closed by doing a "do die" on itself). You can also imagine to run two simulations, and to communicate from one to an other through the experiment, as it is shown in this easy model, where agents can move from one simulation to another :
model smallWorld
global {
int grid_size <- 10;
bool modelleft <- true;
int id<- 0;
int nb_agents <- 50;
init {
create people number: nb_agents {
my_cell <- one_of(cell);
location <- my_cell.location;
}
if (modelleft) {
ask cell where (each.grid_x = (grid_size - 1)) {
color <- #red;
}
} else {
ask cell where (each.grid_x = 0) {
color <- #red;
}
}
}
action changeWorld(rgb color, point loc) {
create people with:[color::color, location::loc] {
my_cell <- cell(location);
}
}
}
species people {
rgb color <- rnd_color(255);
cell my_cell;
reflex move {
if (modelleft and my_cell.color = #red) {
ask smallWorld_model[1] {
do changeWorld(myself.color, {100 - myself.location.x,myself.location.y});
}
do die;
} else {
list<cell> free_cells <- list<cell> (my_cell.neighbors) where empty(people inside each);
if not empty(free_cells) {
my_cell <- one_of(free_cells);
location <- my_cell.location;
}
}
}
aspect default {
draw circle(50/grid_size) color: color;
}
}
grid cell width: grid_size height: grid_size;
experiment fromWorldToWorld type: gui {
init {
create simulation with:[grid_size::20, modelleft::false, id::1, nb_agents::0];
}
output {
display map {
grid cell lines: #black;
species people;
}
}
}
Here is an other example of application of application, available in the model library. Here we run 4 times the Ant Foraging model, with different parameters.
Random seed​
Defining the seed from the model​
If you run several simulations, you may want to use the same seed for each one of those simulations (to compare the influence of a certain parameter, in exactly the same conditions).
Let's remind you that seed
is a built-in attribute of the model. You than just need to specify the value of your seed during the creation of the simulation if you want to fix the seed :
create simulation with:[seed::10.0];
You can also specify the seed if you are inside the init
scope of your global
agent.
global {
init {
seed<-10.0;
}
}
Notice that if you affect the value of your seed built-in directly in the global scope, the affectation of the parameters (for instance specified with the facet with
of the statement create
), and the "init" will be done after will be done at the end.
Defining the seed from the experiment​
The experiment agent also have a built-in attribute seed
. The value of this seed is defined in your simulation preferences. The first simulation created is created with the seed value of the experiment.
The following sequence diagram can explain you better how the affectation of the seed attribute works :
The affectation of an attribute is always done in this order :
(1) the attribute is affected with a specific value in the species scope. If no attribute value is specified, the value is a default value.
(2) if a value is specified for this attribute in the create
statement, then the attribute value is affected again.
(3) the attribute value can be changed again in the init
scope.
Run several simulations with the same random numbers​
The following code shows how to run several simulations with a specific seed, determined from the experiment agent :
model multi_simulations
global {
init {
create my_species;
}
}
species my_species skills:[moving] {
reflex update {
do wander;
}
aspect base {
draw circle(2) color:#green;
}
}
experiment my_experiment type:gui {
float seedValue <- 10.0;
float seed <- seedValue; // force the value of the seed.
init {
// create a second simulation with the same seed as the main simulation
create simulation with:[seed::seedValue];
}
output {
display my_display {
species my_species aspect:base;
}
}
}
When you run this simulation, their execution is exactly similar.
Let's try now to add a new species in this model, and to add a parameter to the simulation for the number of agents created for this species.
model multi_simulations
global {
int number_of_speciesB <- 1;
init {
create my_speciesA;
create my_speciesB number:number_of_speciesB;
}
}
species my_speciesA skills:[moving] {
reflex update {
do wander;
}
aspect base {
draw circle(2) color:#green;
}
}
species my_speciesB skills:[moving] {
reflex update {
do wander;
}
aspect base {
draw circle(2) color:#red;
}
}
experiment my_experiment type:gui {
float seedValue <- 10.0;
float seed <- seedValue; // force the value of the seed.
init {
create simulation with:[seed::seedValue,number_of_speciesB::2];
}
output {
display my_display {
species my_speciesA aspect:base;
species my_speciesB aspect:base;
}
}
}
Then you run the experiment, you may find something strange...
Even if the first step seems ok (the greed agent and one of the two red agent is initialized with the same location), the simulation differs completly. You should have expected to have the same behavior for the greed agent in both of the simulation, but it is not the case. The explaination of this behavior is that a random number generator has generated more random numbers in the second simulation than in the first one.
If you don't understand, here is a short example that may help you to understand better :
model multi_simulations
global {
int iteration_number <- 1;
reflex update {
float value;
loop times:iteration_number {
value<-rnd(10.0);
write value;
}
write "cycle "+cycle+" in experiment "+name+" : "+value;
}
}
experiment my_experiment type:gui {
float seedValue <- 10.0;
float seed <- seedValue; // force the value of the seed.
init {
create simulation with:[seed::seedValue,iteration_number::2];
}
output {
}
}
The output will be something like that :
7.67003069780383
cycle 0 in experiment multi_simulations_model0 : 7.67003069780383
7.67003069780383
0.22889843360303863
cycle 0 in experiment multi_simulations_model1 : 0.22889843360303863
0.22889843360303863
cycle 1 in experiment multi_simulations_model0 : 0.22889843360303863
4.5220913306263855
0.8363180333035425
cycle 1 in experiment multi_simulations_model1 : 0.8363180333035425
4.5220913306263855
cycle 2 in experiment multi_simulations_model0 : 4.5220913306263855
5.460148568140819
4.158355846617511
cycle 2 in experiment multi_simulations_model1 : 4.158355846617511
0.8363180333035425
cycle 3 in experiment multi_simulations_model0 : 0.8363180333035425
1.886091659169562
4.371253083874633
cycle 3 in experiment multi_simulations_model1 : 4.371253083874633
Which means :
Cycle | Value generated in simulation 0 | Value generated in simulation 1 |
---|---|---|
1 | 7.67003069780383 | 7.67003069780383 |
0.22889843360303863 | ||
2 | 0.22889843360303863 | 4.5220913306263855 |
0.8363180333035425 | ||
3 | 4.5220913306263855 | 5.460148568140819 |
4.158355846617511 |
When writing your models, you have to be aware of this behavior. Remember that each simulation has it's own random number generator.
Change the RNG​
The RNG (random number generator) can also be changed : rng
is a string built-in attribute of the experiment (and also of the model). You can choose among the following rng :
- mersenne (by default)
- cellular
- java
The following model shows how to run 4 simulations with the same seed but with some different RNG :
model multi_simulations
global {
init {
create my_species number:50;
}
}
species my_species skills:[moving] {
reflex update {
do wander;
}
aspect base {
draw square(2) color:#blue;
}
}
experiment my_experiment type:gui {
float seed <- 10.0;
init {
create simulation with:[rng::"cellular",seed::10.0];
create simulation with:[rng::"java",seed::10.0];
}
output {
display my_display {
species my_species aspect:base;
graphics "my_graphic" {
draw rectangle(35,10) at:{0,0} color:#lightgrey;
draw rng at:{3,3} font:font("Helvetica", 20 , #plain) color:#black;
}
}
}
}