// // DLASystem.cpp // #include "DLASystem.h" using std::cout; using std::endl; // this function gets called every step, // if there is an active particle then it gets moved, // if not then add a particle void DLASystem::Update() { this->frame++; if (lastParticleIsActive == 1) { moveLastParticle(); } else if (this->particleList.size() < (size_t) endNum) { addParticleOnAddCircle(); setParticleActive(); } else { this->running = false; } } void DLASystem::clearParticles() { // delete particles and the particle list for (size_t i = 0; i < this->particleList.size(); i++) { delete particleList[i]; } particleList.clear(); } // remove any existing particles and setup initial condition void DLASystem::Reset() { // stop running running = false; frame = 0; clearParticles(); lastParticleIsActive = 0; // set the grid to zero for (int i = 0; i < gridSize; i++) { for (int j = 0; j < gridSize; j++) { grid[i][j] = 0; } } // setup initial condition and parameters addCircle = 10; killCircle = 2.0 * addCircle; clusterRadius = 0.0; // add a single particle at the origin double pos[] = {0.0, 0.0}; addParticle(pos); } // set the value of a grid cell for a particular position // note the position has the initial particle at (0,0) // but this corresponds to the middle of the grid array ie grid[ halfGrid ][ halfGrid ] void DLASystem::setGrid(double pos[], int val) { int halfGrid = gridSize / 2; grid[(int) (pos[0] + halfGrid)][(int) (pos[1] + halfGrid)] = val; } // read the grid cell for a given position int DLASystem::readGrid(double pos[]) { int halfGrid = gridSize / 2; return grid[(int) (pos[0] + halfGrid)][(int) (pos[1] + halfGrid)]; } // add a particle to the system at a specific position void DLASystem::addParticle(double pos[]) { // create a new particle Particle *p = new Particle(pos); // push_back means "add this to the end of the list" particleList.push_back(p); // pos coordinates should be -gridSize/2 < x < gridSize/2 setGrid(pos, 1); } // add a particle to the system at a random position on the addCircle // if we hit an occupied site then we do nothing except print a message // (this should never happen) void DLASystem::addParticleOnAddCircle() { double pos[2]; double theta = rgen.random01() * 2 * M_PI; pos[0] = ceil(addCircle * cos(theta)); pos[1] = ceil(addCircle * sin(theta)); if (readGrid(pos) == 0) { addParticle(pos); } else { cout << "FAIL " << pos[0] << " " << pos[1] << endl; } } // send back the position of a neighbour of a given grid cell // NOTE: there is no check that the neighbour is inside the grid, // this has to be done separately... void DLASystem::setPosNeighbour(double setpos[], double pos[], int val) { switch (val) { case 0: setpos[0] = pos[0] + 1.0; setpos[1] = pos[1]; break; case 1: setpos[0] = pos[0] - 1.0; setpos[1] = pos[1]; break; case 2: setpos[0] = pos[0]; setpos[1] = pos[1] + 1.0; break; case 3: setpos[0] = pos[0]; setpos[1] = pos[1] - 1.0; break; } } // when we add a particle to the cluster, we should update the cluster radius // and the sizes of the addCircle and the killCircle void DLASystem::updateClusterRadius(double pos[]) { double newRadius = distanceFromOrigin(pos); if (newRadius > clusterRadius) { clusterRadius = newRadius; // this is how big addCircle is supposed to be: // either 20% more than cluster radius, or at least 5 bigger. double check = clusterRadius * addRatio; if (check < clusterRadius + 5) { check = clusterRadius + 5; } // if it is smaller then update everything... if (addCircle < check) { addCircle = check; killCircle = killRatio * addCircle; } if (killCircle + 2 >= gridSize / 2) { this->running = false; } } } // make a random move of the last particle in the particleList void DLASystem::moveLastParticle() { int rr = rgen.randomInt(4); // pick a random number in the range 0-3, which direction do we hop? double newpos[2]; Particle *lastP = particleList[this->particleList.size() - 1]; setPosNeighbour(newpos, lastP->pos, rr); if (distanceFromOrigin(newpos) > killCircle) { setGrid(lastP->pos, 0); particleList.pop_back(); // remove particle from particleList setParticleInactive(); } else if (readGrid(newpos) == 0) { setGrid(lastP->pos, 0); // set the old grid site to empty // update the position particleList[this->particleList.size() - 1]->pos[0] = newpos[0]; particleList[this->particleList.size() - 1]->pos[1] = newpos[1]; setGrid(lastP->pos, 1); // set the new grid site to be occupied // check if we stick if (checkStick()) { this->csv_out << frame << "," << lastP->pos[0] << "," << lastP->pos[1] << endl; setParticleInactive(); // make the particle inactive (stuck) updateClusterRadius(lastP->pos); // update the cluster radius, addCircle, etc. } } else { // if we get to here then we are trying to move to an occupied site // (this should never happen as long as the sticking probability is 1.0) cout << "reject " << rr << endl; cout << lastP->pos[0] << " " << lastP->pos[1] << endl; } } // check if the last particle should stick (to a neighbour) int DLASystem::checkStick() { Particle *lastP = particleList[this->particleList.size() - 1]; int result = 0; // loop over neighbours for (int i = 0; i < 4; i++) { double checkpos[2]; setPosNeighbour(checkpos, lastP->pos, i); // if the neighbour is occupied... if (readGrid(checkpos) == 1) result = 1; } return result; } // constructor DLASystem::DLASystem(std::ofstream csv_out) { cout << "creating system, gridSize " << gridSize << endl; endNum = 1000; this->csv_out = std::move(csv_out); // allocate memory for the grid, remember to free the memory in destructor grid = new int *[gridSize]; for (int i = 0; i < gridSize; i++) { grid[i] = new int[gridSize]; } // reset initial parameters Reset(); addRatio = 1.2; // how much bigger the addCircle should be, compared to cluster radius killRatio = 1.7; // how much bigger is the killCircle, compared to the addCircle // this opens a logfile, if we want to... //logfile.open("opfile.txt"); } // destructor DLASystem::~DLASystem() { // delete the particles clearParticles(); // delete the grid for (int i = 0; i < gridSize; i++) delete[] grid[i]; delete[] grid; if (csv_out.is_open()) { csv_out.close(); } }