DM: um primeiro programa: mudanças entre as edições

De Física Computacional
Ir para navegação Ir para pesquisar
Sem resumo de edição
(Desfeita a edição 353 de Heitor (Discussão))
 
(32 revisões intermediárias por 2 usuários não estão sendo mostradas)
Linha 22: Linha 22:
#define Lx 16
#define Lx 16
#define Ly 16
#define Ly 16
#define dt 0.01
 
#define Rc 4
#define Rc 4
#define Rc2 (Rc*Rc)
#define Rc2 (Rc*Rc)
#define T 1.
#define T 1.
#define transiente 1000
 
#define medir 10
#define dt 0.01
#define transient 5.
#define total_time 10.
#define measure 0.1
#define samples ((total_time-transient)/measure)
</pre>
</pre>


Linha 38: Linha 42:
<math> dt </math> define o intervalo de tempo de integração na simulação.
<math> dt </math> define o intervalo de tempo de integração na simulação.


<math> R_c </math> é o raio acima do qual o cálculo da força é desconsiderado.  
<math> R_c </math> é o raio de corte acima do qual o cálculo da força é desconsiderado.  
 
<math> total\_time </math> é o tempo absoluto da simulação.
 
<math> transient </math> é o tempo que esperamos até o sistema supostamente equilibrar e, então, começarmos a fazer medidas.
 
<math> measure </math> define o intervalo de tempo entre as medições. 
 
<math> samples </math> nos dá o número total de medidas que serão realizadas
 


Conforme o programa é construído, novas constantes podem eventualmente aparecer à medida que novas funções são implementadas. Abaixo declaramos as funções essenciais ao programa, onde cada uma delas será detalhadamente explicada no decorrer da implementação.  
Conforme o programa é construído, novas constantes podem eventualmente aparecer à medida que novas funções são implementadas. Abaixo declaramos as funções essenciais ao programa, onde cada uma delas será detalhadamente explicada no decorrer da implementação.  
Linha 54: Linha 67:
void main(void)
void main(void)
{
{
FILE *file1,*file2,*file3,*file4;
 
FILE *file1,*file2;
file1=fopen("energy.dat","w");
file1=fopen("energy.dat","w");
file2=fopen("trajectories.dat","w");
file2=fopen("trajectories.dat","w");


  /*********** declaring variables    ************/
unsigned i;
  // declara as variaveis necessarias
double *x,*y,*xo,*yo,*vx,*vy,*Fx,*Fy,Ep,Ec,Vxm=0.,t;
  // ex: double *fx ;
 
size_t size=N*sizeof(double);
 
x= (double*)malloc(size);
y= (double*)malloc(size);
xo=(double*)malloc(size);
yo=(double*)malloc(size);
vx=(double*)malloc(size);
vy=(double*)malloc(size);
Fx=(double*)malloc(size);
Fy=(double*)malloc(size);


</pre>
</pre>


  /**********  allocating memory    ************/
Acima são declaradas todas as variáveis essenciais ao programa, assim como a alocação de memória para os vetores como posição, velocidade, força, etc.  
  // aloca a memoria
Com os vetores alocados, podemos agora inicializar as velocidades e posições das partículas na rede
  // ex. de alocacao DINAMICA de memoria:
 
  // fx  = (double *)malloc( NP * sizeof(double) ); 
<pre>
 
initialize_velocities(vx,vy);
  initialize_positions( px, py);
initialize_positions(x,y,xo,yo,vx,vy);
  initialize_velocities( vx, vy);
</pre>
  for(i = 0; i < NP; i++)
    { ppx[i] = px[i] - vx[i]*deltat;  ppy[i] = py[i] - vy[i]*deltat; } // positions in the initial time
 
  tt=0;  iframe=0;
  // faz um loop no tempo
  //      chama a funcao que calcula forcas
  //      chama a funcao que integra as Eqs de movimento
  //   incrementa o tempo de simulacao
  //      a cada intervalo de tempo definido, fazer medidas


Abaixo é mostrada a parte principal do programa, o loop sobre o tempo. A cada passo de tempo, a rotina <math> compute\_forces </math> calcula a força e a energia potencial das partículas a partir de suas posições, na sequência, a rotina <math> integrate\_EqMotion </math> atualiza as posições das partículas. O <math> for </math> em <math> i </math> imprime no arquivo "trajectories.dat" as posições e velocidades de todas as partículas no instante <math> t </math>. No <math> if </math> testamos se o tempo total é maior que o tempo de equilíbrio e se a medida deve ser feita, caso seja, no arquivo "energy.dat" imprimiremos, respectivamente, o tempo, energia cinética, energia potencial e a energia total.


<pre>
for(t=0.;t<total_time;t+=dt)
{
        compute_forces(Fx,Fy,x,y,&Ep);
        integrate_EqMotion(Fx,Fy,x,y,xo,yo,&Ec,&Vxm);
        for(i=0;i<N;i++)fprintf(file2,"%lf %lf %lf %lf\n",x[i],y[i],vx[i],vy[i]);
        fflush(file2);
        rewind(file2);
        if( t>transient && (1.-fmod(t/measure,1.) )<0.00001 )
        {
                fprintf(file1,"%lf %lf %lf %lf\n",t,Ec,Ep,Ec+Ep);
                fflush(file1);
        }
}
}


fclose(file1);
fclose(file2);


void initialize_positions( double *xx, double *yy )
free(x);
free(y);
free(xo);
free(yo);
free(vx);
free(vy);
free(Fx);
free(Fy);
}
</pre>
 
Agora descrevemos detalhadamente todas as rotinas que aparecem na parte principal. Começamos com a inicialização das velocidades, a rotina recebe os ponteiros das velocidades nas direções <math> x </math> e <math> y </math>. Como condição inicial, todas as partículas recebem uma velocidade aleatória dada por rand()/RAND_MAX que gera um número entre zero e um. Vcx e Vcy são, respectivamente, as componentes da velocidade do centro de massa do sistema nas direções <math> x </math> e <math> y </math>, enquanto V2 é a energia cinética. Para o caso em que as partículas estejam transladando em uma direção preferencial, descontamos suas velocidades pela do centro de massa, além disso, as renormalizamos por um fator de escala que depende da temperatura inicial dada, lembrando que a mesma pode variar no ensamble microcanônico. 
<pre>
 
void initialize_velocities(double *vx,double *vy)
{
{
   int      i ;
   unsigned i;
   double   deltax, deltay;  
   double Vcx=0.,Vcy=0.,V2=0.,fs;
    
 
   deltax = (double)Lx/Nxdeltay = (double)Ly/Ny;
   for(i=0;i<N;i++)
   for(i = 0; i < NP; i++)
   {
    {
      vx[i]=1.*rand()/RAND_MAX;
      xx[i] = (i%Nx)*deltax ;  
      vy[i]=1.*rand()/RAND_MAX;
      yy[i] = (i/Ny)*deltay ;
 
      // printf("%lf %lf   %d \n", xx[i], yy[i], i);
      Vcx+=vx[i];
    }
      Vcy+=vy[i];
      V2 +=vx[i]*vx[i] + vy[i]*vy[i];
  }
 
  Vcx/=N;
  Vcy/=N;
   V2 /=N;
  fs=sqrt(2.*T/V2);
 
   for(i=0;i<N;i++)
  {
        vx[i]=(vx[i]-Vcx)*fs;
        vy[i]=(vy[i]-Vcy)*fs;
   }
 
}
}
</pre>


void initialize_velocities(double *vx, double *vy )
{
  int      i ;
  double  vcmx, vcmy, v2, fs;


  vcmx=0.0;  vcmy=0.0;  v2=0.0;
  for(i = 0; i < NP; i++)
    {
      vx[i] = (double)rand()/RAND_MAX ;
      vy[i] = (double)rand()/RAND_MAX ;
      vcmx += vx[i];    vcmy += vy[i];
      v2 += vx[i]*vx[i] + vy[i]*vy[i] ;  // kinetic energy
    }
  vcmx /= NP ;  vcmy /= NP ;  v2 /= NP;
  fs=sqrt(2*TEMP/v2) ;  // fator de escala -- escolhe uma dada temp T e computa isto : SE escolher uma dist maxwelliana, isto nao eh necessario (me parece)
 
  // set the velocity center of mass to zero (linear momentum of the system is zero)
  // and set its value in such a way that \sum m_i v^2 = Kb T for each degree of freedom
  for(i = 0; i < NP; i++)
    {  vx[i] = (vx[i] - vcmx)*fs ;        vy[i] = (vy[i]-vcmy)*fs ;  }
}


// fx = -dV/dx = -(x/r) (dV/dr)
A rotina da inicialização das posições recebe os ponteiros das coordenadas <math> x </math> e <math> y </math> das partículas, as coordenadas no tempo anterior <math> x_0 </math> e <math> y_0 </math>, além das velocidades inicias <math> v_x </math> e <math> v_y </math>.  Podemos imaginar a disposição inicial das partículas como uma rede quadrada representada na Figura 2, onde em cada vértice se encontra uma partícula, a distância entre elas é dada por <math> dx=L_x/N_x </math> e <math> dy=L_y/N_y </math>. Como <math> i </math> é o índice global das partículas, o resto (%) da divisão inteira entre <math> L_x </math> e <math> Nx </math> nos dará a coordenada <math> x </math>, enquanto o resultado da divisão inteira entre <math> L_y </math> e <math> Ny </math> definirá a coordenada <math> y </math>.
// fx= 48x/r² (1/r^12 - 1/r^6) --  LJ in reduced Units
 
void compute_forces( double *ffx, double *ffy, double *xx, double *yy, double *eenergia )  
[[Arquivo:initialcondition.jpg|thumb|300px|Figura 2 - Condição inicial.]]
 
<pre>
void initialize_positions(double *x,double *y,double *xo,double *yo,double *vx,double *vy)
{
{
  int    i, j;
 
  double x1, y1, x2, y2,dxx, dyy, r2, r2i, r6, ff, energia=0.0 ;
        unsigned i;
 
        double dx,dy;
  for( i=0; i < NP; i++ )
 
    {  ffx[i] = 0.0;   ffy[i] = 0.0; }
        dx=(double)Lx/Nx;
   
        dy=(double)Ly/Ny;
  for( i=0; i < NP; i++ )
 
    {
        for(i=0;i<N;i++)
      x1  = xx[i];    y1 = yy[i];
        {
    for( j=i+1; j < NP; j++ )
                x[i]=(i%Nx)*dx;
{
                y[i]=(i/Ny)*dy;
  x2  = xx[j];    y2 = yy[j];
 
 
                xo[i]=x[i]-vx[i]*dt;
  dxx = x1 - x2;          // compute the distance between particles taking into account the periodic boundary conditions 
                yo[i]=y[i]-vy[i]*dt;
  dxx = fmod(dxx, Lx); 
        }
  dxx=dxx-rint(dxx/Lx)*Lx ;   //note: rint(arg) : rounds its argument to the nearest whole number
 
 
  dyy = y1 - y2;         
  dyy = fmod(dyy, Ly);
  dyy=dyy-rint(dyy/Ly)*Ly ;
 
  r2 = dxx*dxx + dyy*dyy ;
  if(r2<RC2) // rc2: critical value or r
    {
      r2i = 1/r2 ;       
      r6 = r2i*r2i*r2i;
      ff = 48*r2i*r6*(r6-0.5);                        // LJ potential integrated                       
      ffx[i] += ff* dxx ;   ffy[i] += ff* dyy  ;      // update forces
      ffx[j] -= ff* dxx ;  ffy[j] -= ff* dyy  ;
      energia += 4*r6*(r6-1) ;                        // compute energy
    }
}
    }
  *eenergia=energia;
}
}
</pre>
O cálculo da força é a parte mais crucial do programa, pois é a que mais consome recursos e pode ser drasticamente melhorada com otimizações. Abaixo introduzimos a rotina mais simples possível para o cálculo da força. Ela recebe os ponteiros das forças nas direções <math> x </math> e <math> y </math>, as coordenadas (o potencial Lennard-Jones só depende da posição) e a energia potencial. No primeiro loop zeramos as duas componentes da força para todas as partículas.
<pre>
void compute_forces(double *fx,double *fy,double *x,double *y,double *Ep)
{
  unsigned i,j;
  double  x1,y1,x2,y2,dx,dy,r2,r2i,r6,ff,Energy=0.;


  for(i=0;i<N;i++)
  {
        fx[i] = 0.;
        fy[i] = 0.;
  }
</pre>


void integrate_EqMotion( double *ffx, double *ffy, double *xx, double *yy, double *xxp, double *yyp, double eenergia, double *energy_pp, double *vxm )  
Como a aceleração resultante sobre uma partícula é a soma de todas as forças que agem sobre ela, para cada uma delas teríamos que fazer um loop sobre todas as outras, porém, pela Terceira Lei de Newton <math> F_{ij}=-F_{ji} </math>, ou seja, apenas temos que somar sobre todas as combinações de pares de partículas. No primeiro loop abaixo varremos <math> i </math> para todas partículas, no segundo, a varredura começa em <math> i+1 </math> e segue até <math> N-1 </math>. Desse modo, asseguramos que todos os pares de partículas entrem no cálculo, no caso, temos <math> N(N-1)/2 </math> pares. <math> x_1 </math> e <math> x_2 </math> são, respectivamente, as posições das partículas <math> i </math> e <math> j </math>, enquanto <math> dx </math> e <math> dy </math> as distâncias entre elas. Porém, como nosso sistema possui condições periódicas de contorno, nenhum par de partículas poderá ter um distância maior que <math> L/2 </math> em uma dada direção, além disso, elas não precisam necessariamente estar limitadas aos contornos da caixa as quais foram inseridas, mas o cálculo da força sim. Desse modo, a função resto fmod(dx,Lx) nos assegura que nenhuma distância estará fora da caixa. No próximo passo, a função rint(dx/Lx) arredondará o valor da divisão para o inteiro mais próximo, assim, se <math> dx/L_x > 0.5 </math> descontaremos <math> L_x </math> de <math> dx </math> e se <math> dx/L_x \le 0.5 </math> já teremos o <math> dx </math> correto. O mesmo procedimento também é aplicado sobre a componente <math> y </math>. No ''if'' testamos se a distância ao quadrado entre um par de partículas é menor que o raio crítico de interação, ou seja, a distância máxima acima da qual desconsideramos a força entre duas partículas, lembrando que isso é feito devido ao fato do potencial Lennard-Jones ser de curto alcance. 
{
 
   int    i;  
<pre>
   double  xp, yp, vxi, vyi,  vcmx, vcmy, v2, temp  ;
   for(i=0;i<N;i++)
   {
        x1=x[i]; y1=y[i];
 
        for(j=i+1;j<N;j++)
        {
                x2=x[j];
 
                dx=x1 - x2;
                dx=fmod(dx,Lx);
                dx=dx-rint(dx/Lx)*Lx;
 
                y2=y[j];
 
                dy=y1 - y2;
                dy=fmod(dy,Ly);
                dy=dy-rint(dy/Ly)*Ly;
 
                r2=dx*dx + dy*dy;


  vcmx=0.0;  vcmy=0.0;  v2=0.0;
                if(r2<Rc2)
  for(i = 0; i < NP; i++)
                {
    {
                        r2i=1/r2 ;
      xp = 2*xx[i] - xxp[i] + deltat2 * ffx[i] ;
                        r6=r2i*r2i*r2i;
      yp = 2*yy[i] - yyp[i] + deltat2 * ffy[i] ;
                        ff=48*r2i*r6*(r6-0.5);
      vxi = (xp - xx[i]) / (2*deltat);
                        fx[i] += ff*dxfy[i] += ff*dy;
      vyi = (yp - yy[i]) / (2*deltat);
                        fx[j] -= ff*dx;   fy[j] -= ff*dy;
     
                         Energy += 4*r6*(r6-1);
      vcmx += vxi;    vcmy += vyi;        // to monitor the average velocity
            }
      v2 += vxi*vxi + vyi*vyi ;            // kinetic energy
        }
      xxp[i] = xx[i] ;  yyp[i] = yy[i] // update the "previous position"
  }
      yyp[i] = yy[i] ;  
   *Ep=Energy;
      xx[i] = xp;                         // update the positions in a current time
      yy[i] = yp;
  }
  temp=v2/(2*NP) ;                         // kbT = < m*v^2 > per degree of freedom
  *energy_pp=(eenergia + 0.5*v2)/NP;       // energia total = ( U + K ) -- energy per particle =energia/NP
   *vxm = vcmx /NP ;
}
}
</pre>

Edição atual tal como às 18h36min de 16 de setembro de 2015

[EM CONSTRUÇÃO] Aqui mostraremos o passo a passo de como montar o primeiro programa para simulação de dinâmica molecular. A linguagem de programação utilizada será o C. Como um primeiro esboço, o programa pode ser organizado como mostra o fluxograma abaixo. Posteriormente, introduziremos partes do código comentando cada etapa. Para o leitor mais familiarizado com DM, ao final da página há o programa completo sem comentários.


A ideia básica da simulação pode ser entendida como: Leitura dos parâmetros da simulação, inicialização das posições das partículas, inicialização das velocidades das partículas, laço temporal onde se calcula a força, integra-se as equações de movimento e, de tempos em tempos, medidas são realizadas.

Fluxograma.jpg

Abaixo introduzimos partes do código onde começamos declarando as bibliotecas standard utilizadas do C

#include<stdio.h>
#include<stdlib.h>
#include<math.h>

Está parte é reservada para declarações de constantes que definem os parâmetros da simulação. As constantes podem ser definidas utilizando-se das diretivas de compilação do C através do comando "#define" ou podem ser lidas a partir de um arquivo externo chamado pelo programa. A vantagem do segundo método é que o programa só precisará ser compilado uma única vez para diferentes condições iniciais. Porém, por simplicidade e clareza, optaremos pelo primeiro caso.

#define Nx 16
#define Ny 16
#define Np (Nx*Ny)
#define Lx 16
#define Ly 16

#define Rc 4
#define Rc2 (Rc*Rc)
#define T 1.

#define dt 0.01
#define transient 5.
#define total_time 10.
#define measure 0.1
#define samples ((total_time-transient)/measure)

representam, respectivamente, o número de partículas nas direções .

representam, respectivamente, o tamanho da rede nas direções .

número total de partículas

define o intervalo de tempo de integração na simulação.

é o raio de corte acima do qual o cálculo da força é desconsiderado.

é o tempo absoluto da simulação.

é o tempo que esperamos até o sistema supostamente equilibrar e, então, começarmos a fazer medidas.

define o intervalo de tempo entre as medições.

nos dá o número total de medidas que serão realizadas


Conforme o programa é construído, novas constantes podem eventualmente aparecer à medida que novas funções são implementadas. Abaixo declaramos as funções essenciais ao programa, onde cada uma delas será detalhadamente explicada no decorrer da implementação.

void initialize_positions(double *xx, double *yy ) ;
void initialize_velocities(double *vx, double *vy ) ;
void compute_forces( double *ffx, double *ffy, double *xx, double *yy, double *eenergia ) ;
void integrate_EqMotion( double *fx, double *fy, double *xx, double *yy,  double *xxp, double *yyp, double eenergia, double *energy_pp, double *vxm ) ;

A seguir introduzimos a função principal

void main(void)
{

FILE *file1,*file2;
file1=fopen("energy.dat","w");
file2=fopen("trajectories.dat","w");

unsigned i;
double *x,*y,*xo,*yo,*vx,*vy,*Fx,*Fy,Ep,Ec,Vxm=0.,t;

size_t size=N*sizeof(double);

x= (double*)malloc(size);
y= (double*)malloc(size);
xo=(double*)malloc(size);
yo=(double*)malloc(size);
vx=(double*)malloc(size);
vy=(double*)malloc(size);
Fx=(double*)malloc(size);
Fy=(double*)malloc(size);

Acima são declaradas todas as variáveis essenciais ao programa, assim como a alocação de memória para os vetores como posição, velocidade, força, etc. Com os vetores alocados, podemos agora inicializar as velocidades e posições das partículas na rede

initialize_velocities(vx,vy);
initialize_positions(x,y,xo,yo,vx,vy);

Abaixo é mostrada a parte principal do programa, o loop sobre o tempo. A cada passo de tempo, a rotina calcula a força e a energia potencial das partículas a partir de suas posições, na sequência, a rotina atualiza as posições das partículas. O em imprime no arquivo "trajectories.dat" as posições e velocidades de todas as partículas no instante . No testamos se o tempo total é maior que o tempo de equilíbrio e se a medida deve ser feita, caso seja, no arquivo "energy.dat" imprimiremos, respectivamente, o tempo, energia cinética, energia potencial e a energia total.

for(t=0.;t<total_time;t+=dt)
{
        compute_forces(Fx,Fy,x,y,&Ep);
        integrate_EqMotion(Fx,Fy,x,y,xo,yo,&Ec,&Vxm);

        for(i=0;i<N;i++)fprintf(file2,"%lf %lf %lf %lf\n",x[i],y[i],vx[i],vy[i]);
        fflush(file2);
        rewind(file2);

        if( t>transient && (1.-fmod(t/measure,1.) )<0.00001 )
        {
                fprintf(file1,"%lf %lf %lf %lf\n",t,Ec,Ep,Ec+Ep);
                fflush(file1);
        }
}

fclose(file1);
fclose(file2);

free(x);
free(y);
free(xo);
free(yo);
free(vx);
free(vy);
free(Fx);
free(Fy);
}

Agora descrevemos detalhadamente todas as rotinas que aparecem na parte principal. Começamos com a inicialização das velocidades, a rotina recebe os ponteiros das velocidades nas direções e . Como condição inicial, todas as partículas recebem uma velocidade aleatória dada por rand()/RAND_MAX que gera um número entre zero e um. Vcx e Vcy são, respectivamente, as componentes da velocidade do centro de massa do sistema nas direções e , enquanto V2 é a energia cinética. Para o caso em que as partículas estejam transladando em uma direção preferencial, descontamos suas velocidades pela do centro de massa, além disso, as renormalizamos por um fator de escala que depende da temperatura inicial dada, lembrando que a mesma pode variar no ensamble microcanônico.


void initialize_velocities(double *vx,double *vy)
{
  unsigned i;
  double Vcx=0.,Vcy=0.,V2=0.,fs;

  for(i=0;i<N;i++)
  {
      vx[i]=1.*rand()/RAND_MAX;
      vy[i]=1.*rand()/RAND_MAX;

      Vcx+=vx[i];
      Vcy+=vy[i];
      V2 +=vx[i]*vx[i] + vy[i]*vy[i];
  }

  Vcx/=N;
  Vcy/=N;
  V2 /=N;
  fs=sqrt(2.*T/V2);

  for(i=0;i<N;i++)
  {
        vx[i]=(vx[i]-Vcx)*fs;
        vy[i]=(vy[i]-Vcy)*fs;
  }

}


A rotina da inicialização das posições recebe os ponteiros das coordenadas e das partículas, as coordenadas no tempo anterior e , além das velocidades inicias e . Podemos imaginar a disposição inicial das partículas como uma rede quadrada representada na Figura 2, onde em cada vértice se encontra uma partícula, a distância entre elas é dada por e . Como é o índice global das partículas, o resto (%) da divisão inteira entre e nos dará a coordenada , enquanto o resultado da divisão inteira entre e definirá a coordenada .

Figura 2 - Condição inicial.
void initialize_positions(double *x,double *y,double *xo,double *yo,double *vx,double *vy)
{

        unsigned i;
        double dx,dy;

        dx=(double)Lx/Nx;
        dy=(double)Ly/Ny;

        for(i=0;i<N;i++)
        {
                x[i]=(i%Nx)*dx;
                y[i]=(i/Ny)*dy;

                xo[i]=x[i]-vx[i]*dt;
                yo[i]=y[i]-vy[i]*dt;
        }

}

O cálculo da força é a parte mais crucial do programa, pois é a que mais consome recursos e pode ser drasticamente melhorada com otimizações. Abaixo introduzimos a rotina mais simples possível para o cálculo da força. Ela recebe os ponteiros das forças nas direções e , as coordenadas (o potencial Lennard-Jones só depende da posição) e a energia potencial. No primeiro loop zeramos as duas componentes da força para todas as partículas.

void compute_forces(double *fx,double *fy,double *x,double *y,double *Ep)
{
  unsigned i,j;
  double  x1,y1,x2,y2,dx,dy,r2,r2i,r6,ff,Energy=0.;

  for(i=0;i<N;i++)
  {
        fx[i] = 0.;
        fy[i] = 0.;
  }

Como a aceleração resultante sobre uma partícula é a soma de todas as forças que agem sobre ela, para cada uma delas teríamos que fazer um loop sobre todas as outras, porém, pela Terceira Lei de Newton , ou seja, apenas temos que somar sobre todas as combinações de pares de partículas. No primeiro loop abaixo varremos para todas partículas, no segundo, a varredura começa em e segue até . Desse modo, asseguramos que todos os pares de partículas entrem no cálculo, no caso, temos pares. e são, respectivamente, as posições das partículas e , enquanto e as distâncias entre elas. Porém, como nosso sistema possui condições periódicas de contorno, nenhum par de partículas poderá ter um distância maior que em uma dada direção, além disso, elas não precisam necessariamente estar limitadas aos contornos da caixa as quais foram inseridas, mas o cálculo da força sim. Desse modo, a função resto fmod(dx,Lx) nos assegura que nenhuma distância estará fora da caixa. No próximo passo, a função rint(dx/Lx) arredondará o valor da divisão para o inteiro mais próximo, assim, se descontaremos de e se já teremos o correto. O mesmo procedimento também é aplicado sobre a componente . No if testamos se a distância ao quadrado entre um par de partículas é menor que o raio crítico de interação, ou seja, a distância máxima acima da qual desconsideramos a força entre duas partículas, lembrando que isso é feito devido ao fato do potencial Lennard-Jones ser de curto alcance.

  for(i=0;i<N;i++)
  {
        x1=x[i]; y1=y[i];

        for(j=i+1;j<N;j++)
        {
                x2=x[j];

                dx=x1 - x2;
                dx=fmod(dx,Lx);
                dx=dx-rint(dx/Lx)*Lx;

                y2=y[j];

                dy=y1 - y2;
                dy=fmod(dy,Ly);
                dy=dy-rint(dy/Ly)*Ly;

                r2=dx*dx + dy*dy;

                if(r2<Rc2)
                {
                        r2i=1/r2 ;
                        r6=r2i*r2i*r2i;
                        ff=48*r2i*r6*(r6-0.5);
                        fx[i] += ff*dx;   fy[i] += ff*dy;
                        fx[j] -= ff*dx;   fy[j] -= ff*dy;
                        Energy += 4*r6*(r6-1);
            }
        }
  }
  *Ep=Energy;
}