Next:ЗАКЛЮЧЕНИЕ
Up:8. ПРИМЕРЫ ПРОГРАММ
Prev:8.2. Перемножение матриц

8.3. Решение краевой задачи методом Якоби

Еще одна область задач, для которых достаточно хорошо разработана технология параллельного программирования - это краевые задачи. В этом случае используется техника декомпозиции по процессорам расчетной области, как правило, с перекрытием подобластей. На Рис. 22 представлено такое разложение исходной расчетной области на 4 процессора с топологией одномерной сетки. Заштрихованные области на каждом процессоре обозначают те точки, в которых расчет не производится, но они необходимы для выполнения расчета в пограничных точках. В них должна быть предварительно помещена информация с пограничного слоя соседнего процессора.

Рис. 22. Пример декомпозиции двумерной расчетной области

Приведем пример программы решения уравнения Лапласа методом Якоби на двумерной регулярной сетке. В программе декомпозиция области выполнена не по строкам, как на рисунке, а по столбцам. Так удобнее при программировании на языке FORTRAN, на C удобнее разбиение производить по строкам (это определяется способом размещения матриц в памяти компьютера).

С Программа решения уравнения Лапласа методом Якоби с
С использованием функции MPI_Sendrecv и NULL процессов
    PROGRAM JACOBI
    IMPLICIT NONE
    INCLUDE 'mpif.h'
    INTEGER n,m,npmin,nps,itmax
C параметры:
С n      - количество точек области в каждом направлении;
С npmin  - минимальное число процессоров для решения задачи;
С npsаа  - число столбцов локальной части матрицы. Этот параметр
С          введен в целях экономии памяти;
С itmax  - максимальное число итераций, если сходимость не будет
С          достигнута.
    PARAMETER (n = 400, npmin=1, nps=n/npmin+1, itmax = 1000)
    REAL*8 A(0:n+1,0:nps+1), B(n,nps)
    REAL*8 diffnorm, gdiffnorm, eps, time
    INTEGER left, right, i, j, k, itcnt, status(0:3), tag
    INTEGER IAM, NPROCS, ierr
    LOGICAL converged
С определение числа процессоров, выделенных задаче (NPROCS),
C и номера текущего процессора (IAM)
    CALL MPI_INIT(IERR)
    CALL MPI_COMM_SIZE(MPI_COMM_WORLD, NPROCS, ierr)
    CALL MPI_COMM_RANK(MPI_COMM_WORLD, IAM, ierr)
С установка критерия достижения сходимости
    eps = 0.01
С вычисление числа столбцов, обрабатываемых процессором
    m = n/NPROCS
    IF (IAM.LT.(n-NPROCS*m)) THEN
    m = m+1
    END IF
    time = MPI_Wtime()
С задание начальных и граничных значений
    do j = 0,m+1
    do i = 0,n+1
     a(i,j) = 0.0
    end do
    end do

    do j = 0,m+1
    A(0,j) = 1.0
    A(n+1,j) = 1.0
    end do
    IF(IAM.EQ.0) then
    do i = 0,n+1
    A(i,0) = 1.0
    end do
    end if
    IF(IAM.EQ.NPROCS-1) then
    do i = 0,n+1
    A(i,m+1) = 1.0
    end do
    end if
С определение номеров процессоров слева и справа. Если таковые
С отсутствуют, то им присваивается значение MPI_PROC_NULL
С (для них коммуникационные операцииа игнорируются)
    IF (IAM.EQ.0) THEN
    left = MPI_PROC_NULL
    ELSE
    left = IAM - 1
    END +IF
    IF (IAM.EQ.NPROCS-1) THEN
    right = MPI_PROC_NULL
    ELSE
    right = IAM+1
    END IF
    tag = 100
    itcnt = 0
    converged = .FALSE.
С цикл по итерациям
    DO k = 1,itmax
    diffnorm = 0.0
    itcnt = itcnt + 1
С вычисление новых значений функции по 5-точечной схеме
    DO j = 1, m
    DO i = 1, n
    B(i,j)=0.25*(A(i-1,j)+A(i+1,j)+A(i,j-1)+A(i,j+1))
    diffnorm = diffnorm + (B(i,j)-A(i,j))**2
    END DO
    END DO
С переприсваивание внутренних точек области
    DO j = 1, m
    DO i = 1, n
    A(i,j) = B(i,j)
    END DO
    END DO

С пересылка граничных значений в соседние процессоры
    CALL MPI_SENDRECV(B(1,1), n, MPI_DOUBLE_PRECISION, left, tag,
    $ A(1,0), n, MPI_DOUBLE_PRECISION, left, tag, MPI_COMM_WORLD,
    $ status, ierr)

    CALL MPI_SENDRECV(B(1,m), n, MPI_DOUBLE_PRECISION, right,
    $ tag, A(1,m+1), n, MPI_DOUBLE_PRECISION, right, tag,
    $ MPI_COMM_WORLD, status, ierr)
С вычисление невязки и проверка условия достижения сходимости
    CALL MPI_Allreduce( diffnorm, gdiffnorm, 1, MPI_DOUBLE_PRECISION,
    $ MPI_SUM, MPI_COMM_WORLD, ierr )
    gdiffnorm = sqrt( gdiffnorm)
    converged = gdiffnorm.LT.eps
    if(converged) goto 777
    END DO
 777 continue

    time = MPI_Wtime() - time
    IF(IAM.EQ.0) then
    WRITE(*,*) ' At iteration ', itcnt, 'а diff is ', gdiffnorm
    WRITE(*,*) ' Time calculation: ', time
    END IF   
    CALL MPI_Finalize(ierr)
    stop
    end



Next:ЗАКЛЮЧЕНИЕ
Up:8. ПРИМЕРЫ ПРОГРАММ
Prev:8.2. Перемножение матриц