Loading [MathJax]/jax/output/CommonHTML/jax.js
본문 바로가기
수학의 재미/아름다운 이론

FDM #7, Heat Equation의 풀이(3)

by hustler78 2022. 8. 1.
728x90
반응형

 

이 글은

2022.08.01 - [수학의 재미/아름다운 이론] - FDM #6, Heat Equation의 풀이(2)

 

FDM #6, Heat Equation의 풀이(2)

이 글은 2022.07.30 - [수학의 재미/아름다운 이론] - FDM #5, Heat Equation의 풀이(1) FDM #5, Heat Equation의 풀이(1) 이번 글은 2022.07.25 - [수학의 재미/아름다운 이론] - 미분방정식을 연립방정식으로, F..

sine-qua-none.tistory.com

에서 이어집니다.

 

지난 글의 똑같은 문제를 다른 방식의 FDM으로 접근해 보겠습니다.

 

열방정식의 예시(Revisited)

저번 글에서 다루었던 예시는 다음과 같습니다.

  ut(t,x)=uxx(t,x),0<x<1,t>0,

○ 초기 조건: u(0,x)=sin(πx)
○ 경계조건: u(t,0)=u(t,1)=0
 
※ 참고 이 방정식의 exact solution 은
u(t,x)=sin(πx)exp(π2t)

 

이산화 과정(지난 글과 똑같으므로 건너뛰어도 됨)

저번 글에서 알아보았듯이 시점 변수 t와 변위 변수 x를 다음과 같이 격자로 나눕니다.

열방정식의 이산화 과정

1. 시간 t 축을 유한개의 균등 격자로 나눔
  - 식(1)에는 초기시점 t=0은 있지만, 끝시점은 없죠. 따라서 관심 있는 충분히 큰 끝시점 T를 하나 정하고,
  0=t0<t1<t2<<tN1<tN=T
    로 균등하게 나눔
  - 각 시점의 차이는 k로 일정함. 즉, tn+1tn=k

2. 변위 변수 x도 유한개의 균등 격자로 나눔
 - 식(1)의 x가 움직이는 구간을 [xm,xM]이라 했을 때,
xm=x0<x1<x2<<xJ1<xJ=xM
   으로 균등하게 나눔
- 각 변위의 차이는 h로 일정함. 즉, xj+1xj=h 

이제 (t,x)를 격자를 사용하여 편미분 방정식을 연립방정식으로 만들어 보겠습니다.

 

 

FDM 설계

이제 (t,x)를 격자를 사용하여 편미분 방정식을 연립방정식으로 만들어 보겠습니다. 

0=t0<t1<<tn=T

xm=x0<x1<<xJ=xM

에 대해 식(1)을 아래의 방법으로 이산화 시킵니다. t 수열 사이 간격은 k, x수열 사이 간격은 h입니다.

 

시점 tn에서 backward difference 방식으로 이산화 시키자!

그러면 점 (tn+1,xj)에서 k 만큼의 시점 차로 backward difference를 구하면

 

ut(tn+1,xj)=u(tn+1,xj)u(tn+1k,xj)k이고

tn+1k=tn입니다.

 

저번 글과 마찬가지로 unj=u(tn,xj) 라 정의하면

 

ut(tn+1,xj)=un+1junjk

 

이죠. x에 대해서는 central difference를 사용합니다. x의 차이를 h라 하면

 

uxx(tn+1,xj)=u(tn+1,xj+h)2u(tn+1,xj)+u(tn+1,xjh)h2

입니다. xj±h=xj±1 이므로 준식은

uxx(tn+1,xj)=un+1j+12un+1j+un+1j1h2

 

따라서 ut(t,x)=uxx(t,x)는 식(2), (3)을 결합하여 다음과 같이 변합니다.

un+1junjk=un+1j+12un+1j+un+1j1h2

 

α=kh2 이라 하면


un+1junj=α(un+1j+12un+1j+un+1j1)

정리하면

unj=αun+1j1+(1+2α)un+1jαun+1j+1

 

이걸 어떻게 해석해야 할까요?

n+1 step에 있는 j1,j,j+1 번째 값 3개를 가지고 n step의 j번째 값을 구할 수 있다???

위의 해석은 터무니없죠. 어떻게 미래 시점인 n+1의 정보를 가지고 과거 시점인 n을 구할 수 있나요. 따라서 다른 식으로 해석해 봅시다.

 

  • n step상의 j번째 점 하나가 n+1 step j1,  j, j+1 번째 점에 영향을 미칩니다.(그림의 투명한 테트리스 블록 참조) 또한
  • n step상의 j1번째 점 하나가 n+1 step j2,j1,  j 번째 점에 영향을 미칩니다.
  • n step상의 j+1번째 점 하나가 n+1step  jj+1,j+2  번째 점에 영향을 미칩니다.

 

역으로 생각하면, 

n+1 step의 j번째 값에 영향을 미치는 것은 n step의 j1,j,j+1번째 점

입니다. 좀 더 자세히 써봅시다.

 

 

식(*)은

j=1일 때  un1=αun+10+(1+2α)un+11αun+12=(1+2α)un+11αun+12
1<j<J1 일 때, unj=αun+1j1+(1+2α)un+1jαun+1j+1
j=J1일 때, unJ1=αun+1J2+(1+2α)un+1J1αun+1J=αun+1J2+(1+2α)un+1J1

 

라 쓸 수 있고, 마지막 단계로 위의 관계식을 행렬로 표현해보겠습니다.

 


(1+2ααα1+2ααα1+2ααα1+2α)(un+11un+12un+1J2un+1J1)=(un1un2unJ2unJ1)

 

이 행렬 방정식의 행렬들을 차례대로 T,˜un+1,˜un 이라 하면

T˜un+1=˜un

입니다. 

이제 FDM 푸는 절차를 얘기해보죠.

 

 

FDM 절차

아래와 같은 과정으로 모든 격자점에서의 u의 값을 산출합니다.

1. u0j 산출, 즉 t=0에서의 값 산출

초기 조건을 이용하여 u0j=sin(πxj) 로 구함

 

2. u1j 값 산출

u0j에서 양 끝점을 뺀 것이 ˜u0이고, 식(4)에서

˜u1=T1˜u0 

으로  구하거나 아니면 T 가 tridiagonal matrix 이므로 3중 대각행렬의 풀이법인 Thomas Algorithm을 사용하여 ˜u1 산출. 여기에 경계조건 u10=u1J=0을 반영하여 온전한 형태의 u1j을 모두 얻음

 

3. unj 에서 un+1j 값 모두 산출

unj을 다 알고 있으므로 여기서 양 끝점을 뺀 ˜un 을 구하고 2번과 같이 T의 역행렬이나 Thomas Algorithm을 사용하여 ˜un+1j 을 얻음. 여기에 경계조건 un+10=un+1J=0을 첨가하여 un+1j을 모두 얻음

 

4. uNj 까지 3번을 반복하여 모두 얻음

 

 

 

FDM 과정을 알아봤으니 직접 코딩하여 문제를 해결해 보겠습니다.

 

 

 

 

FDM 코드

python으로 작성한 위 문제의 FDM 풀이법은 다음과 같습니다.

def Solve_HeatEqaution_1D_implicit():
    # u_t(t,x) = u_xx(t,x) 0<x<1, t>0
    # u(0,x) = sin(πx)
    # u(t,0)= u(t,1) =0
    # closed form solution : u(t,x)= sin(πx)exp(-π^2 t)

    x_min, x_max = 0, 1
    t0 = 0
    maturity = 1

    num_x, num_t = 50, 10000
    # num_x, num_t = 50, 4500
    # num_x, num_t = 70, 10000
    # num_x, num_t = 80, 10000

    x_seq = np.linspace(x_min, x_max, num_x + 1)
    t_seq = np.linspace(t0, maturity, num_t + 1)

    h = x_seq[1] - x_seq[0]
    k = t_seq[1] - t_seq[0]

    alpha = k / h ** 2

    print('delta of x : {:.3f}'.format(k))
    print('delta of t : {:.3f}'.format(h))
    print('alpha (= k/(h^2)) : {:.3f}'.format(alpha))

    u = np.empty((num_t + 1, num_x + 1))

    # u(t,x_min) =0
    u[:, 0] = 0

    # u(t, x_max) =0
    u[:, -1] = 0

    # u(0, x) = sin( pi*x)
    u[0, :] = np.sin(np.pi * x_seq)

    # construct coeff matrix : num_x -1 Size
    diag = np.ones(num_x - 1) * (1 + 2 * alpha)
    under = np.ones(num_x - 1) * (-alpha)
    over = np.ones(num_x - 1) * (-alpha)

    for n in range(num_t):
        known = u[n, 1:num_x]
        unknown = thomas(under, diag, over, known)
        u[n + 1, 1:num_x] = unknown

    t_idx = -1
    real_t = t_seq[t_idx]

    sol = np.sin(np.pi * x_seq) * np.exp(-np.pi * np.pi * real_t)
    fdm_sol = u[t_idx, :]

    max_of_difference = np.max(sol - fdm_sol)
    print('max distance between exact solution and fdm solution is {:.4f}'.format(max_of_difference))

    fig, axes = plt.subplots(1, 2, figsize=(10, 20))
    axes[0].plot(x_seq, sol, 'b', label='exact solution')
    axes[0].plot(x_seq, fdm_sol, 'm', label='fdm solution')
    axes[0].set_title(' u(x) when k={:.3f}, h={:.3f}, alpha={:.3f}'.format(k, h, alpha))
    axes[0].set_xlabel('x')
    axes[0].set_ylabel('u(x)')
    axes[0].legend()

    axes[1].plot(x_seq, sol - fdm_sol, 'r')
    axes[1].set_xlabel('x')
    axes[1].set_ylabel('u(x)')
    axes[1].set_title('difference between exact and fdm')

    plt.show()
    
# Function for TMDA Algorithm
def thomas(a, b, c, d):
    """ A is the tridiagnonal coefficient matrix and d is the RHS matrix"""
    """
    a is lower diagonal a2,a3,..,a_N, meaning
    b is diagonal b1,b2,b3,..,b_N meaning
    c is upper diagonal c1,c2,c3,.. c_{N-1} meaning
    """
    N = len(a)
    cp = np.zeros(N, dtype='float64')  # store tranformed c or c'
    dp = np.zeros(N, dtype='float64')  # store transformed d or d'
    X = np.zeros(N, dtype='float64')  # store unknown coefficients

    # Perform Forward Sweep
    # Equation 1 indexed as 0 in python
    cp[0] = c[0] / b[0]
    dp[0] = d[0] / b[0]
    # Equation 2, ..., N (indexed 1 - N-1 in Python)
    for i in np.arange(1, (N), 1):
        dnum = b[i] - a[i] * cp[i - 1]
        cp[i] = c[i] / dnum
        dp[i] = (d[i] - a[i] * dp[i - 1]) / dnum

    # Perform Back Substitution
    X[(N - 1)] = dp[N - 1]  # Obtain last xn

    for i in np.arange((N - 2), -1, -1):  # use x[i+1] to obtain x[i]
        X[i] = (dp[i]) - (cp[i]) * (X[i + 1])

    return (X)


if __name__ == '__main__':
    Solve_HeatEqaution_1D_implicit()

 

간략히 살펴볼까요. 저번 글과 비슷한 부분들은 모두 생략합니다. (보다 자세한 설명은 여기를 참조). Thomas algorithm 부분도 생략합니다.

 

    # construct coeff matrix : num_x -1 Size
    diag = np.ones(num_x - 1) * (1 + 2 * alpha)	# tridiagonal 의 주대각선을 설계
    under = np.ones(num_x - 1) * (-alpha)	# tridiagoal의 주대각선 아래대각선
    over = np.ones(num_x - 1) * (-alpha)	# tridiagoal의 주대각선 위대각선

    for n in range(num_t):
        known = u[n, 1:num_x]			#u의 양 끝점을 뺀 뒤
        unknown = thomas(under, diag, over, known)	#thomas algorithm 적용

 

 

결과

다양한 격자에 대해 계산한 결과는 아래와 같습니다. exact solution과 FDM 결과의 차이 그래프도 수록합니다.

1. num_x, num_t = 50, 10000

 

2. num_x, num_t = 50, 4500

 

3. num_x, num_t = 70, 10000

 

4. num_x, num_t = 80, 10000

 

 

결론

FDM을 시점에 대해 backward difference로 놓았더니 다양한 격자에 대해 모두 좋은 결과가 나오는군요.

 

α가 0.5보다 작은 것 여부에 상관없이 모든 경우에 대해 결과가 수렴합니다.  이렇게 bakward difference를 사용하여 FDM를 구성하고 푸는 방법을

함축적 방법(Implicit method)

이라고 합니다.

 

안정성 측면에서 implicit method를 사용하는 것이 좋으나, 위에서 보다시피 계산량이 많습니다. 행렬 방정식을 풀어야 하기 때문이지. 뿐만 아니고 3중대각행렬을 푸는 Thomas Algorithm도 곁들여 알아놓아야 합니다.

 

이상으로 Heat equation를 FDM으로 푸는 두 가지 방법인 explicit method와 implicit method를 모두 알아봤습니다. 이 heat equation 은 금융공학에서도 만나게 되는데, 파생상품의 가치가 만족하는 방정식이 바로 heat equation으로 나타나기 때문이죠.

 

이른바 Black Scholes Equation에 대해 다룰 때, 다시 이 글을 소환해 보도록 하겠습니다.

728x90
반응형

댓글