ChinaNet
标题: [转帖]wolff算法的C代码 [打印本页]
作者: Doublen 时间: 2015-4-8 10:22
标题: [转帖]wolff算法的C代码
[size=13.63636302947998px]/* Wolff single-cluster algorithm for the two dimensional nearest neighbor
Ising model By Jian-Sheng Wang, February 1995.
Compile with
[size=13.63636302947998px] cc -O Wolff.c -lm
-O for optimization, and -lm to link the math library.
*/
[size=13.63636302947998px]#include <stdio.h>
#include <stdlib.h>
#include <math.h>
/* macro definitions */
#define L 16 /* lattice linear size */
#define N (L*L) /* total number of spins */
#define Z 4 /* coordination number = 2*d */
#define MCTOT 500 /* total Monte Carlo steps */
#define MCDIS 200 /* steps discarded in the beginning */
/* global variables */
int s[N]; /* spin +1 or -1 */
double p = 1.0; /* percolation probability */
[size=13.63636302947998px] /* funcition prototypes */
void neighbor(int i, int nn[ ]);
void flip(int i, int s0);
void monte_carlo_steps(int n);
void energy(double *);
[size=13.63636302947998px]/* The main program, running the Monte Carlo loop, collecting data */
[size=13.63636302947998px]void main()
{
int i, mc;
double e = 0;
[size=13.63636302947998px] p = 1 - exp( - 2/2.269);
for (i = 0; i < N; ++i) /* initialize, all spin up */
s = 1;
[size=13.63636302947998px] for(mc = 0; mc < MCTOT; ++ mc) {
monte_carlo_steps(5);
if( mc >= MCDIS)
energy(&e);
}
printf("<e> = %f\n", e/(MCTOT-MCDIS)/N);
}
[size=13.63636302947998px]/* This function monte_carlo_steps performs n cluster flips, eqivalent to
few Monte Carlo steps in standard single-spin-flip algorithms.
It picks a seed site at random and calls the flip function to generate
one cluster. N is total number of spin. They are macro definitions. */
[size=13.63636302947998px]void monte_carlo_steps(int n)
{
int i, k;
[size=13.63636302947998px] for(k = 0; k < n; ++k) {
i = drand48() * (double) N;
flip(i, s);
}
}
[size=13.63636302947998px]/* Perform a Wolff single cluster flip. s[], p, and Z are passed globally.
The first argument i of flip function is the site to be flipped, the
second argument is the spin of the cluster before flipping. */
[size=13.63636302947998px]void flip(int i, int s0)
{
int j, nn[Z];
[size=13.63636302947998px] s = - s0; /* flip the spin immediately */
neighbor(i, nn); /* find nearest neighbor of i */
for(j = 0; j < Z; ++j) /* flip the neighbor if ... */
if(s0 == s[nn[j]] && drand48() < p)
flip(nn[j], s0);
}
[size=13.63636302947998px]/* Neighbor returns in the array nn[ ] the neighbor sites of i. The sites
are labelled sequentially, starting from 0. It works for any hypercubic
lattice. Z (=2*D) is the coordination number, passed as a macro defintion.
L is linear size, also passed as a macro definition. */
[size=13.63636302947998px]void neighbor(int i, int nn[ ])
{
int j, r, p, q;
[size=13.63636302947998px] r = i;
p = 1 - L;
q = 1;
[size=13.63636302947998px] for(j = 0; j < Z; j += 2) {
nn[j] = (r + 1) % L == 0 ? i + p : i + q;
nn[j+1] = r % L == 0 ? i - p : i - q;
r = r/L;
p *= L;
q *= L;
}
}
[size=13.63636302947998px]/* This function calculate the energy of the configuration s[],
in fact, it is the negative of the energy. */
[size=13.63636302947998px]void energy(double *e)
{
int i, j, ie = 0;
int nn[Z];
for(i = 0; i < N; ++i) {
neighbor(i, nn); /* find the neighbor of center site */
for(j = 0; j < Z; j += 2) /* look at positive direction only */
ie += s*s[nn[j]]; /* add nearest neighbor interaction */
}
*e += ie; /* accumulant energy */
}
[size=13.63636302947998px]其中drand48() 为0-1之间均匀分布的随机数
[size=13.63636302947998px]本帖转自52MC
欢迎光临 ChinaNet (http://www.nuclear.net.cn/) |
Powered by Discuz! X3.1 |