Processing math: 31%
 

『ゼロからできるMCMC』 Chapter7 Ising modelをJuliaで

『ゼロからできるMCMC』 Chapter7 Ising model(イジングモデル)のコードを, 練習を兼ねてJuliaで書いてみました. 私は, イジングモデルに関しては全く素人なので結果の解釈等については大きな錯誤があるかもしれません. また,詳細なことを書くと著作権的にだめな気もするので, 解説などはありません.

Version 1.6.2

Metropolis法によるサンプリング

Julia code

N×N 格子で温度は T, 結合定数は J, 外部磁場は h としている. 初期状態はランダムに+1, -1を各格子に与えた. 格子点 ix, iyをランダムに選び, 確率 min でその格子 ( s_{i_{x}, i_{y}} ) のスピンを反転させる. \Delta E は,

\begin{align} \large \Delta E = 2 s_{i_{x}, i_{y}} \left( J \sum_{\pm 1} (s_{i_{x} \pm 1, i_{y}} + s_{i_{x}, i_{y} \pm 1}) + h \right) \tag{7.50} \end{align}

と計算した. 計算回数はNiterステップに一回のサンプリングとした. そして, Nprint回サンプルした.

境界条件は周期境界条件とした. 端っこに対応するために, nm, npという変数をつくっている. これは例えば N = 5 の 場合だと,

  • nm = [5, 1, 2, 3, 4]
  • np = [2, 3, 4, 5, 1]
という配列である.

Julia
  1. using Plots
  2. using Random:seed!
  3.  
  4. ## パラメータ
  5. struct Params
  6. N::Int64
  7. Niter::Int64
  8. J::Float64
  9. h::Float64
  10. T::Float64
  11. Nprint::Int64
  12. end
  13.  
  14. ## 初期配位
  15. function initializeState(N)
  16. # s = ones(N, N);
  17. s = randn(N, N);
  18. @. s[s>0] = 1
  19. @. s[s<0] = -1
  20. return s
  21. end
  22.  
  23. ## ΔEの計算
  24. function calcdE(s, J, h, ix, iy, np, nm)
  25. dE = 2*s[ix, iy] * (J*(s[np[ix], iy] + s[nm[ix], iy] + s[ix, np[iy]] + s[ix, nm[iy]])+h);
  26. return dE
  27. end
  28.  
  29. ## Metropolis法で更新
  30. function isingMetropolis(p::Params)
  31.  
  32. naccept = 0;
  33. s = zeros(p.N, p.N, p.Nprint);
  34. s[:, :, 1] = initializeState(p.N);
  35. np = [Vector(2:p.N); 1]
  36. nm = [p.N; Vector(1:p.N-1)];
  37.  
  38. for I = 2:p.Nprint
  39. ss = s[:, :, I-1];
  40. for K = 1:p.Niter
  41.  
  42. # select a random cell
  43. ix = rand(1:p.N);
  44. iy = rand(1:p.N);
  45.  
  46. # calculate action
  47. dE = calcdE(ss, p.J, p.h, ix, iy, np, nm);
  48.  
  49. # Metropolis test
  50. if exp(-dE/p.T) > rand()
  51. ss[ix, iy] = -ss[ix, iy];
  52. naccept += 1;
  53. end
  54. end
  55. s[:, :, I] = ss;
  56. end
  57. r = 100*naccept/(p.Nprint*p.Niter);
  58. display("acceptance rate is $(r) %")
  59. return s
  60. end
  61.  
  62.  
  63. ## ここからメイン
  64. p = Params(100, #N
  65. 100^2*10, # Niter
  66. 1.0, # J
  67. 0.0, # h
  68. 2/log(1+sqrt(2)), #T (crit: 2/log(1+sqrt(2)))
  69. 100) # Nprint;
  70. @time S = isingMetropolis(p::Params);
  71.  
  72. # gifとして保存
  73. anim = Animation()
  74. for I = 2:p.Nprint
  75. fig = heatmap(S[:, :, I],
  76. aspect_ratio=:equal,
  77. showaxis=:false,
  78. grid=:false,
  79. ticks = nothing,
  80. size = (200, 200),
  81. colorbar=:none)
  82. frame(anim, fig)
  83. end
  84. gif(anim, "ising3.gif", fps=10)

コード内に示したような設定で計算した. 0.454313 seconds (2.93 k allocations: 15.539 MiB)とのこと. 温度は相転移温度に設定した.

name

教科書には臨界減速の話が載っているので, このケースについても自己相関をみてみた. 荒いサンプリングで申し訳ないが, 相転移温度に設定したことによって強い自己相関が現れているのがわかる.

name
Juliaコード Julia
  1. spinSum = [sum(S[:, :, I]) for I in 1:p.Nprint]
  2. plt =scatter(1:p.Nprint, spinSum)
  3. xlabel!("smaple (per 10^5 steps)")
  4. ylabel!("The sum of the spin")
  5. plt[1][:legend] = :none
  6. plt[:dpi] = 300;
  7. png(plt, "ising4")

外部磁場の変化

結合定数 J = 1, 温度 T=1 に固定し, 外部磁場 h を変化させてみる. 格子サイズは64, 40960ステップに一回のサンプリングとする. これは, 教科書図7.9の再現である.

初期状態はすべてのスピンが上向き ( +1 ) とする. そして, 初期の外部磁場は h = -0.4 で 100サンプルに一回磁場を 0.1下げ, 400回のサンプリングを行なう. なお, 外部磁場が負のときの基底状態はすべてのスピンが下向きである.

コードは, 磁場を下げるたびに ( 100回サンプルするたびに ) そのタイムステップまでの100回分のスピンの合計を計算している. また, そのあと前の100回の最後の状態を次の初期状態としないといけないので, 新たに上の節とは異なる函数をつくった. 絶対もっと効率的な方法があると思うが, Juliaは速いので気にならなかった.

Julia
  1. function initializeState(S)
  2. s = S[:, :, end]
  3. return s
  4. end
  5.  
  6. # 前の基底状態を引き継げるようにした函数
  7. function isingMetropolisContinue(p::Params, S)
  8.  
  9. naccept = 0;
  10. s = zeros(p.N, p.N, p.Nprint);
  11. s[:, :, 1] = initializeState(S);
  12. np = [Vector(2:p.N); 1]
  13. nm = [p.N; Vector(1:p.N-1)];
  14.  
  15. for I = 2:p.Nprint
  16. ss = s[:, :, I-1];
  17. for K = 1:p.Niter
  18.  
  19. # select a random cell
  20. ix = rand(1:p.N);
  21. iy = rand(1:p.N);
  22.  
  23. # calculate action
  24. dE = calcdE(ss, p.J, p.h, ix, iy, np, nm);
  25.  
  26. # Metropolis test
  27. if exp(-dE/p.T) > rand()
  28. ss[ix, iy] = -ss[ix, iy];
  29. naccept += 1;
  30. end
  31. end
  32. s[:, :, I] = ss;
  33. end
  34. r = 100*naccept/(p.Nprint*p.Niter);
  35. display("acceptance rate is $(r) %")
  36. return s
  37. end
  38.  
  39. # ここからがメイン
  40. sumSpin = zeros(1, 400)
  41. hs = [-0.4 -0.5 -0.6 -0.7];
  42. S = ones(64, 64, 1);
  43. for H = 1:4
  44. p = Params(64, 64^2*10, 1.0, hs[H], 1.0, 100)
  45. @time S = isingMetropolisContinue(p::Params, S);
  46. sumSpin[1+100*(H-1):100*H]= [sum(S[:, :, I]) for I = 1:100]
  47. end
  48.  
  49. #plot
  50. sumSpin = sumSpin’;
  51. fig = scatter(1:100, sumSpin[1:100], markershape=:x, label="h=-0.4")
  52. scatter!(101:200, sumSpin[101:200], markershape=:x, label="h=-0.5")
  53. scatter!(201:300, sumSpin[201:300], markershape=:x, label="h=-0.6")
  54. scatter!(301:400, sumSpin[301:400], markershape=:x, label="h=-0.7")
  55. xlabel!("iteration")
  56. ylabel!("The sum of the spins")
  57. png("ising1.img")
  58.  
name

h=-0.4 ではなかなかスピンがすべて1の状態から抜け出せていない. h=-0.6 くらいまで下げてやっと基底状態 ( スピンがすべて−1, 即ちスピンの和は−4096 ) へ遷移し始める.

上の例は T=1.0 でのシミュレーションであった. T=1.5 と少し温度を上げてやると以下のように速やかに基底状態へ遷移する ( この辺の解釈については統計力学を勉強したことがないので自信がない ) .

name

Wolffのアルゴリズム

解説は教科書を見ていただきたい.

Julia
  1. using Plots
  2.  
  3. ## parameters
  4. struct Params
  5. N::Int64
  6. nIter::Int64
  7. J::Float64
  8. T::Float64
  9. nPrint::Int64
  10. end
  11.  
  12. function initializeState(N)
  13. s = randn(N, N);
  14. @. s[s>0] = 1
  15. @. s[s<0] = -1
  16. return s
  17. end
  18.  
  19. function isingWolff(p::Params)
  20. S = zeros(p.N, p.N, p.nPrint) # preallocation
  21. S[:, :, 1] = initializeState(p.N) # 初期状態を格納
  22. np = [Vector(2:p.N); 1] # 1:1:Nに+1した値
  23. nm = [p.N; Vector(1:p.N-1)] # 1:1:Nに-1した値
  24. prob = 1 - exp(-2 * p.J / p.T) # クラスタに追加する確率
  25. s = S[:, :, 1] # 初期状態から開始
  26. for J = 2:p.nPrint # p.nPrint回サンプリング
  27. for I = 1:p.nIter # サンプリング毎にp.nIterステップ ( クラスタの反転 ) 行なう
  28. # クラスタを反転させる函数
  29. s = clusterFlip(prob, nm, np, p.N, s)
  30. end
  31. S[:, :, J] = s
  32. display(J)
  33. end
  34. return S
  35. end
  36.  
  37.  
  38. # クラスタを反転させる函数
  39. function clusterFlip(prob, nm, np, N, s)
  40. inOrOut = falses(N, N) # クラスタに各点が属するかの情報. 0 is out, 1 is out of the cluster ( 教科書と逆! )
  41. ix = rand(1:p.N) # choose a random point
  42. iy = rand(1:p.N)
  43. inOrOut = falses(N, N)
  44. inOrOut[ix, iy] = true # add the point to the cluster
  45. spinCluster = s[ix, iy] # this cluster' spin (+1 or -1)
  46. iCluster = [ix iy] # 1つ目の点をクラスタに追加
  47. k = 1
  48. while k-1 < size(iCluster, 1)  # ここからは教科書のコードとあまりかわらない
  49. ix, iy = iCluster[k, :] # クラスタに属する点を選択
  50. ixp1 = np[ix]
  51. ixm1 = nm[ix]
  52. iyp1 = np[iy]
  53. iym1 = nm[iy]
  54. # 右隣
  55. # 次のif文は, 隣とスピンが同じでかつ, すでにクラスタに入っていない場合, ということ
  56. if s[ixp1, iy] == spinCluster && ~inOrOut[ixp1, iy]
  57. if rand() < prob
  58. iCluster = [iCluster; [ixp1 iy]]
  59. inOrOut[ixp1, iy] = true;
  60. end
  61. end
  62. # 上隣
  63. if s[ixm1, iy] == spinCluster && ~inOrOut[ixm1, iy]
  64. if rand() < prob
  65. iCluster = [iCluster; [ixm1 iy]]
  66. inOrOut[ixm1, iy] = true;
  67. end
  68. end
  69. # 左隣
  70. if s[ix, iyp1] == spinCluster && ~inOrOut[ix, iyp1]
  71. if rand() < prob
  72. iCluster = [iCluster; [ix iyp1]]
  73. inOrOut[ix, iyp1] = true;
  74. end
  75. end
  76. # 下隣
  77. if s[ix, iym1] == spinCluster && ~inOrOut[ix, iym1]
  78. if rand() < prob
  79. iCluster = [iCluster; [ix iym1]]
  80. inOrOut[ix, iym1] = true;
  81. end
  82. end
  83. k += 1
  84. end
  85. @. s[inOrOut] = -1*s[inOrOut] # クラスタに属する点をいっきに反転
  86. return s
  87. end
  88.  
  89.  
  90. ## メイン
  91. p = Params(200, #N
  92. 10, # Niter
  93. 1.0, # J
  94. 2.30, #2/log(1+sqrt(2)), #T (crit: 2/log(1+sqrt(2)))
  95. 350) # Nprint;
  96. @time S = isingWolff(p)
  97.  
  98. # 動画で保存
  99. anim = Animation()
  100. for I = 2:p.nPrint
  101. fig = heatmap(S[:, :, I],
  102. aspect_ratio=:equal,
  103. showaxis=:false,
  104. grid=:false,
  105. ticks = nothing,
  106. size = (200, 200),
  107. colorbar=:none,
  108. title="t = $(I)")
  109. # c = :cividis) # お好きな色に
  110. frame(anim, fig)
  111. end
  112. gif(anim, "test.gif", fps=20)

上のコード内に示した条件での結果. どんどんクラスタが大きくなり, 最後は基底状態をいったりきたりする結果となった. 最後の方になると, 一つのクラスタが大きくなるからか, 計算時間が非常に長くなる. これは, コードが悪いのかそうなって当然なのかどうだろう. tが1進むごとに, 10ステップ進んでいることに注意.

name

この記事のTOP    BACK    TOP