Skip to main content
Version: 1.9.3

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.

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​

Generate simulations on-the-fly​

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).

Communication between simulations​

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 :

resources/images/exploringModel/change_world.png

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;
}
}
}

For more complex communication example you can look into the example models from the network plugin> TCP Server And Client Example .gaml runs a first simulation that will create two servers (one of them being the lead server), then the experiment will create a new simulation with different parameters, and this simulation will itself create two clients, after this, clients and servers from both simulations will communicate together through the TCP protocol. The same behavior can be found in the WebSocket Server And Client Example .gaml model but with the use of websocket instead of TCP.

Going a step further, TCP Teleportation.gaml recreates the same behavior as the Small world example from the previous statement but this time the whole agent is serialized and de-serialized from one simulation to another then sent through TCP. And this time you can chain as many simulation as you want thanks to an experiment parameter. 1 Pong Teleportation (send agent).gaml exhibits the same behavior but uses the MQTT protocol.

Permanent displays​

Here is an other example of application available in the model library under the name Ant Foraging.gaml with the experiment 3 Simulations. In this simulation, we run 3 times the Ant Foraging model, with different parameters and plot in a unique graph the evolution of the total gathered food by each model.

image

This graph is done thanks to the permanent block:

permanent {
display Comparison background: #white {
chart "Food Gathered" type: series {
loop s over: simulations {
data "Food " + int(s) value: s.food_gathered color: s.color marker: false style: line thickness: 5;
}
}
}
}

This block defines a display that is linked to the experiment only, it can thus access all the currently running simulations with the list simulations. From there you can define displays to summarize what's happening in all the simulations, display some statistics on them etc.

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 :

resources/images/exploringModel/sequence_diagram_seed_affectation.png

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.

resources/images/exploringModel/same_simulation_one_agent.png

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...

resources/images/exploringModel/same_simulation_2_species.png

Even if the first step seems ok (the green agent and one of the two red agent is initialized with the same location), the simulation differs completely. You should have expected to have the same behavior for the green agent in both of the simulation, but it is not the case. The explanation 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 :

CycleValue generated in simulation 0Value generated in simulation 1
17.670030697803837.67003069780383
0.22889843360303863
20.228898433603038634.5220913306263855
0.8363180333035425
34.52209133062638555.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;
}
}
}
}