mirror of
https://github.com/BlackLight/nash.git
synced 2024-11-23 20:25:11 +01:00
First commit for nash
This commit is contained in:
commit
57e65981b6
6 changed files with 517 additions and 0 deletions
6
Makefile
Normal file
6
Makefile
Normal file
|
@ -0,0 +1,6 @@
|
|||
all:
|
||||
g++ -o nash main.cpp nash.cpp -g
|
||||
|
||||
clean:
|
||||
rm tags
|
||||
rm nash
|
62
README
Normal file
62
README
Normal file
|
@ -0,0 +1,62 @@
|
|||
This is a small software that simulates the behaviour of a set of individuals
|
||||
placed (for sake of simplicity) over a matrix where each individual can choose
|
||||
between two strategies:
|
||||
|
||||
- Being cooperative with the other individuals
|
||||
- Being in competition with the other individuals
|
||||
|
||||
From a certain starting point, in which each individuals has its own strategy
|
||||
independantly from the one chosen by the others, the algorithm goes on, and a
|
||||
certain individual x changes its strategy if there is a neighbour having a
|
||||
different strategy and a higher "gain" value (i.e. that strategy takes more
|
||||
benefit). The algorithm is repeated as long as an equilibrium point is reached
|
||||
(Nash's equilibrium) in which no individual will change its strategy anymore as
|
||||
no change is convenient anymore. This balance point won't be always reached
|
||||
anyway, and the algorithm recognizes a scenario that has no Nash's equilibrium
|
||||
point when a loop occurs (i.e. when it takes in exam a configuration that was
|
||||
already met).
|
||||
|
||||
Usage of the program: compile simply issuing `make', then start it with
|
||||
./nash <environment_file> <coop_coop_gain> <coop_comp_gain> <comp_coop_gain> <comp_comp_gain>
|
||||
|
||||
where
|
||||
|
||||
- environment_file is a file containing the starting configuration of the
|
||||
environment (an example is included in the file `environment'). In this file the
|
||||
collaborating individuals are marked by a dot `.', and the ones in competition
|
||||
are marked by a `X'. Starting from here you can easily create your environment
|
||||
file;
|
||||
|
||||
- coop_coop_gain: Score gained when the individual x cooperates and its neighbour
|
||||
cooperates as well;
|
||||
|
||||
- coop_comp_gain: Score gained when the individual x cooperates and its neighbour
|
||||
is in competition;
|
||||
|
||||
- comp_coop_gain: Score gained when the individual x is in competition and its
|
||||
neighbour cooperates;
|
||||
|
||||
- comp_comp_gain: Score gained when both the individual x and its neighbour are
|
||||
in competition.
|
||||
|
||||
An example is the problem of the two prisoners:
|
||||
|
||||
- When A doesn't confess and B doesn't confess too, they both get 8 years of
|
||||
prison (coop_coop_gain = -8);
|
||||
|
||||
- When A doesn't confess and B confesses, A gets 20 years of prison and B is
|
||||
free (coop_comp_gain = -20);
|
||||
|
||||
- When A confesses and B doesn't, A is free and B gets 20 years of prison
|
||||
(comp_coop_gain = 0);
|
||||
|
||||
- When both A and B confess, they both get 14 years of prison (comp_comp_gain =
|
||||
-14).
|
||||
|
||||
For running a simulation of the two prisoners' problem over the example
|
||||
environment just type
|
||||
|
||||
./nash environment -8 -20 0 -14
|
||||
|
||||
Have fun.
|
||||
|
11
environment
Normal file
11
environment
Normal file
|
@ -0,0 +1,11 @@
|
|||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . X . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
||||
. . . . . . . . . . .
|
79
main.cpp
Normal file
79
main.cpp
Normal file
|
@ -0,0 +1,79 @@
|
|||
/*
|
||||
* =====================================================================================
|
||||
*
|
||||
* Filename: main.cpp
|
||||
*
|
||||
* Description: Main file for the simulation of a Nash's environment
|
||||
*
|
||||
* Version: 0.1
|
||||
* Created: 01/11/2010 13:23:03
|
||||
* Revision: none
|
||||
* Compiler: gcc
|
||||
*
|
||||
* Author: BlackLight (http://0x00.ath.cx), <blacklight@autistici.org>
|
||||
* Licence: GNU GPL v.3
|
||||
* Company: DO WHAT YOU WANT CAUSE A PIRATE IS FREE, YOU ARE A PIRATE!
|
||||
*
|
||||
* =====================================================================================
|
||||
*/
|
||||
|
||||
#include <iostream>
|
||||
#include <cstdlib>
|
||||
#include "nash.h"
|
||||
|
||||
using std::cout;
|
||||
using std::cerr;
|
||||
using std::endl;
|
||||
|
||||
/**
|
||||
* \brief Main function for the program
|
||||
*/
|
||||
|
||||
int
|
||||
main ( int argc, char *argv[] )
|
||||
{
|
||||
NashEnvironment *nash = NULL;
|
||||
|
||||
if ( argc < 6 )
|
||||
{
|
||||
cerr << "Usage: " << argv[0] << " <environment_file> <coop_coop_gain> <coop_comp_gain> <comp_coop_gain> <comp_comp_gain>" << endl;
|
||||
return 1;
|
||||
}
|
||||
|
||||
nash = new NashEnvironment ( argv[1] );
|
||||
|
||||
nash->init_costs (
|
||||
strtod ( argv[2], NULL ),
|
||||
strtod ( argv[3], NULL ),
|
||||
strtod ( argv[4], NULL ),
|
||||
strtod ( argv[5], NULL )
|
||||
);
|
||||
|
||||
nash->print();
|
||||
|
||||
try
|
||||
{
|
||||
while ( nash->refresh_strategies() )
|
||||
{
|
||||
cout << endl;
|
||||
|
||||
for ( int i=0; i < nash->getCols(); i++ )
|
||||
cout << "==";
|
||||
|
||||
cout << endl;
|
||||
nash->print();
|
||||
}
|
||||
}
|
||||
|
||||
catch ( NashException e )
|
||||
{
|
||||
cerr << endl << e.what() << endl;
|
||||
delete nash;
|
||||
return 1;
|
||||
}
|
||||
|
||||
cout << endl << "Nash's equilibrium configuration found" << endl;
|
||||
delete nash;
|
||||
return 0;
|
||||
} /* ---------- end of function main ---------- */
|
||||
|
259
nash.cpp
Normal file
259
nash.cpp
Normal file
|
@ -0,0 +1,259 @@
|
|||
/*
|
||||
* =====================================================================================
|
||||
*
|
||||
* Filename: nash.cpp
|
||||
*
|
||||
* Description: Simulation of a cooperative-competitive environemnt of individuals
|
||||
* using Nash's models
|
||||
*
|
||||
* Version: 0.1
|
||||
* Created: 31/10/2010 18:35:14
|
||||
* Revision: none
|
||||
* Compiler: gcc
|
||||
*
|
||||
* Author: BlackLight (http://0x00.ath.cx), <blacklight@autistici.org>
|
||||
* Licence: GNU GPL v.3
|
||||
* Company: DO WHAT YOU WANT CAUSE A PIRATE IS FREE, YOU ARE A PIRATE!
|
||||
*
|
||||
* =====================================================================================
|
||||
*/
|
||||
|
||||
#include <iostream>
|
||||
#include <fstream>
|
||||
#include <vector>
|
||||
#include "nash.h"
|
||||
|
||||
using std::cout;
|
||||
using std::endl;
|
||||
using std::vector;
|
||||
using std::ifstream;
|
||||
|
||||
NashEnvironment::NashEnvironment ( int rows, int cols )
|
||||
{
|
||||
this->rows = rows;
|
||||
this->cols = cols;
|
||||
|
||||
for ( int i=0; i < rows; i++ )
|
||||
{
|
||||
vector<nash_strategy> row;
|
||||
|
||||
for ( int j=0; j < cols; j++ )
|
||||
{
|
||||
row.push_back(coop);
|
||||
}
|
||||
|
||||
matrix.push_back(row);
|
||||
}
|
||||
}
|
||||
|
||||
NashEnvironment::NashEnvironment ( const char *file )
|
||||
{
|
||||
vector<nash_strategy> row;
|
||||
char ch;
|
||||
|
||||
rows = 0;
|
||||
cols = 0;
|
||||
|
||||
ifstream in ( file );
|
||||
|
||||
if ( !in )
|
||||
{
|
||||
throw NashException("Could not open the specified file");
|
||||
}
|
||||
|
||||
while ( !in.eof() )
|
||||
{
|
||||
ch = (char) in.get();
|
||||
|
||||
if ( ch == '\n' )
|
||||
{
|
||||
if ( cols == 0 )
|
||||
{
|
||||
cols = row.size();
|
||||
} else {
|
||||
if ( row.size() != 0 && cols != row.size() )
|
||||
{
|
||||
throw NashException("The size of the rows does not match");
|
||||
}
|
||||
}
|
||||
|
||||
if ( row.size() != 0 )
|
||||
{
|
||||
rows++;
|
||||
matrix.push_back( row );
|
||||
row.clear();
|
||||
}
|
||||
} else if ( ch == '.' ) {
|
||||
row.push_back( coop );
|
||||
} else if ( ch == 'X' ) {
|
||||
row.push_back( comp );
|
||||
}
|
||||
}
|
||||
|
||||
if ( !row.empty() )
|
||||
{
|
||||
matrix.push_back( row );
|
||||
row.clear();
|
||||
}
|
||||
|
||||
if ( rows == 0 )
|
||||
{
|
||||
throw NashException("Invalid or empty strategy file");
|
||||
}
|
||||
}
|
||||
|
||||
const int
|
||||
NashEnvironment::getRows()
|
||||
{
|
||||
return rows;
|
||||
}
|
||||
|
||||
const int
|
||||
NashEnvironment::getCols()
|
||||
{
|
||||
return cols;
|
||||
}
|
||||
|
||||
void
|
||||
NashEnvironment::init_costs ( double coop_coop, double coop_comp, double comp_coop, double comp_comp )
|
||||
{
|
||||
costs[coop][coop] = coop_coop;
|
||||
costs[comp][comp] = comp_comp;
|
||||
costs[comp][coop] = comp_coop;
|
||||
costs[coop][comp] = coop_comp;
|
||||
} /* ----- end of function nash_init_costs ----- */
|
||||
|
||||
void
|
||||
NashEnvironment::print ()
|
||||
{
|
||||
for ( int i=0; i < rows; i++ )
|
||||
{
|
||||
for ( int j=0; j < cols; j++ )
|
||||
{
|
||||
cout << (( matrix[i][j] == coop ) ? "." : "X" ) << " ";
|
||||
}
|
||||
|
||||
cout << endl;
|
||||
}
|
||||
} /* ----- end of function nash_print_matrix ----- */
|
||||
|
||||
void
|
||||
NashEnvironment::update_utilities ( vector< vector< double > >& utilities )
|
||||
{
|
||||
for ( int x=0; x < rows; x++ )
|
||||
{
|
||||
for ( int y=0; y < cols; y++ )
|
||||
{
|
||||
utilities[x][y] = 0.0;
|
||||
|
||||
for ( int i = (( x == 0 ) ? x : x-1 ); i <= (( x == rows-1 ) ? x : x+1 ); i++ )
|
||||
{
|
||||
for ( int j = (( y == 0 ) ? y : y-1 ); j <= (( y == cols-1 ) ? y : y+1 ); j++ )
|
||||
{
|
||||
if ( !( x == i && y == j ))
|
||||
{
|
||||
utilities[x][y] += costs [matrix[x][y]] [matrix[i][j]];
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool
|
||||
NashEnvironment::refresh_strategies ()
|
||||
{
|
||||
double best_utility = 0.0;
|
||||
int best_x = 0, best_y = 0;
|
||||
vector< vector<int> > changes;
|
||||
vector< vector<double> > utilities;
|
||||
|
||||
for ( int i=0; i < rows; i++ )
|
||||
{
|
||||
vector<double> row(cols);
|
||||
utilities.push_back(row);
|
||||
}
|
||||
|
||||
update_utilities ( utilities );
|
||||
|
||||
for ( int x=0; x < rows; x++ )
|
||||
{
|
||||
for ( int y=0; y < cols; y++ )
|
||||
{
|
||||
best_x = x;
|
||||
best_y = y;
|
||||
best_utility = utilities[best_x][best_y];
|
||||
|
||||
for ( int i = (( x == 0 ) ? x : x-1 ); i <= (( x == rows-1 ) ? x : x+1 ); i++ )
|
||||
{
|
||||
for ( int j = (( y == 0 ) ? y : y-1 ); j <= (( y == cols-1 ) ? y : y+1 ); j++ )
|
||||
{
|
||||
if ( !( x == i && y == j ))
|
||||
{
|
||||
if ( utilities[i][j] > best_utility )
|
||||
{
|
||||
best_x = i;
|
||||
best_y = j;
|
||||
best_utility = utilities[best_x][best_y];
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ( matrix[best_x][best_y] != matrix[x][y] )
|
||||
{
|
||||
vector<int> pair(2);
|
||||
pair[0] = x;
|
||||
pair[1] = y;
|
||||
changes.push_back (pair);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for ( int i=0; i < changes.size(); i++ )
|
||||
{
|
||||
matrix[changes[i][0]][changes[i][1]] = ( matrix[changes[i][0]][changes[i][1]] == coop ) ? comp : coop;
|
||||
}
|
||||
|
||||
if ( changes.size() > 0 )
|
||||
{
|
||||
if ( check_loop() )
|
||||
{
|
||||
throw NashException ("Loop detected - No Nash's equilibrium is possible for this configuration");
|
||||
}
|
||||
}
|
||||
|
||||
visited_combinations.push_back(matrix);
|
||||
return ( changes.size() > 0 );
|
||||
} /* ----- end of function nash_refresh_strategies ----- */
|
||||
|
||||
|
||||
bool
|
||||
NashEnvironment::check_loop ()
|
||||
{
|
||||
bool found = false;
|
||||
|
||||
for ( int n=0; n < visited_combinations.size(); n++ )
|
||||
{
|
||||
found = true;
|
||||
|
||||
for ( int i=0; i < rows && found; i++ )
|
||||
{
|
||||
for ( int j=0; j < cols && found; j++ )
|
||||
{
|
||||
if ( matrix[i][j] != visited_combinations[n][i][j] )
|
||||
{
|
||||
found = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ( found )
|
||||
{
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
} /* ----- end of function check_loop ----- */
|
||||
|
100
nash.h
Normal file
100
nash.h
Normal file
|
@ -0,0 +1,100 @@
|
|||
/*
|
||||
* =====================================================================================
|
||||
*
|
||||
* Filename: nash.h
|
||||
*
|
||||
* Description: Header file for the simulation of Nash's model
|
||||
*
|
||||
* Version: 0.1
|
||||
* Created: 01/11/2010 13:20:44
|
||||
* Revision: none
|
||||
* Compiler: gcc
|
||||
*
|
||||
* Author: BlackLight (http://0x00.ath.cx), <blacklight@autistici.org>
|
||||
* Licence: GNU GPL v.3
|
||||
* Company: DO WHAT YOU WANT CAUSE A PIRATE IS FREE, YOU ARE A PIRATE!
|
||||
*
|
||||
* =====================================================================================
|
||||
*/
|
||||
|
||||
#include <vector>
|
||||
#include <exception>
|
||||
typedef enum { coop, comp } nash_strategy;
|
||||
|
||||
class NashException : public std::exception
|
||||
{
|
||||
const char *message;
|
||||
|
||||
public:
|
||||
NashException ( const char *m )
|
||||
{
|
||||
message = m;
|
||||
}
|
||||
|
||||
virtual const char* what()
|
||||
{
|
||||
return message;
|
||||
}
|
||||
};
|
||||
|
||||
class NashEnvironment
|
||||
{
|
||||
int rows;
|
||||
int cols;
|
||||
double costs[2][2];
|
||||
std::vector< std::vector< nash_strategy > > matrix;
|
||||
std::vector< std::vector< std::vector< nash_strategy > > > visited_combinations;
|
||||
|
||||
/**
|
||||
* \brief Verify if a certain scenario has already been visited, for avoiding loops in environments without Nash's balance
|
||||
* \return True if the current scenario was already analyzed, false otherwise
|
||||
*/
|
||||
bool check_loop ();
|
||||
|
||||
/**
|
||||
* \brief Update the utility costs on the matrix
|
||||
* \param utilities Reference to the utilities matrix to be updated
|
||||
*/
|
||||
void update_utilities ( std::vector< std::vector<double> >& utilities );
|
||||
|
||||
public:
|
||||
|
||||
/**
|
||||
* \brief Constructor for the class
|
||||
* \param rows Rows of the Nash environment's matrix
|
||||
* \param cols Cols of the Nash environment's matrix
|
||||
*/
|
||||
NashEnvironment ( int rows, int cols );
|
||||
|
||||
/**
|
||||
* \brief Constructor for the class
|
||||
*/
|
||||
NashEnvironment ( const char *file );
|
||||
|
||||
/**
|
||||
* \brief Number of rows in the matrix
|
||||
*/
|
||||
const int getRows();
|
||||
|
||||
/**
|
||||
* \brief Number of cols in the matrix
|
||||
*/
|
||||
const int getCols();
|
||||
|
||||
/**
|
||||
* \brief Initialize the costs of the matrix of strategy costs
|
||||
*/
|
||||
void init_costs ( double coop_coop, double coop_comp, double comp_coop, double comp_comp );
|
||||
|
||||
/**
|
||||
* \brief Print the current status of the matrix of strategies
|
||||
*/
|
||||
void print ();
|
||||
|
||||
/**
|
||||
* \brief Refresh the matrix of the utility values
|
||||
* \return True if there were strategy changes, false otherwise
|
||||
*/
|
||||
bool refresh_strategies ();
|
||||
};
|
||||
|
Loading…
Reference in a new issue