我正在努力並行化一小部分具有循環和調用子程序的代碼。但結果與1個線程或2個線程不一致。我們不得不使用鎖嗎?OPENMP具有1或n個線程的可重複性
!$OMP parallel private(kk,j,i,k1,j1,i1,k2,j2,i2,ic,icm,xx,yy,ydatm),shared(undef,lt,ln,nd,xd,tgrd,ndaym)
thread_id = OMP_GET_THREAD_NUM()
num_threads = OMP_GET_NUM_THREADS()
if(thread_id.eq.0) thread_id=thread_id+1
start_no = (thread_id * ndaym/num_threads);
end_no = ((thread_id + 1) * ndaym/num_threads);
xx(:)=undef
yy(:,:)=undef
!$OMP DO
DO kk=start_no,end_no
do j=1,nlat
do i=1,nlon
ic=0
do k1=kk-nd,kk+nd
k2=k1
if(k2.lt.1)k2=1
if(k2.gt.ndaym)k2=ndaym
do j1=j-lt,j+lt
j2=j1
if(j2.lt.1)j2=1
if(j2.gt.nlat)j2=nlat
do i1=i-ln,i+ln
i2=i1
if(i2.lt.1)i2=1
if(i2.gt.nlon)i2=i2-nlon
ic=ic+1
yy(ic,:)=xdatm(i2,j2,k2,:)
yya(:) = yy(ic,:)
call funcmean(yya,xx(ic),1,nmem,nmem,undef)
if(k1.eq.kk.and.j1.eq.j.and.i1.eq.i) then
icm=ic
call funcsd(yya,1,nmem,nmem,undef,xx(ic),xd)
endif
enddo ! for i1
enddo ! for j1
enddo ! for k1
call funcens(xx,yy,tgrd,nmem,icm,undef,dist)
ydatm(i,j,kk,:)= dist(:)
enddo !for nlon
enddo ! for nlat
ENDDO !kk
!$OMP end do
deallocate(xx,yy)
!$OMP end parallel
任何人都可以提供一些線索嗎?
它在我看來好像你正在試圖計算每個線程的外部do循環迭代的份額,以及所有那些計算'start_no'和'end_no'的東西。如果是這樣,你肯定錯過了關於OpenMP的一個要點,那就是編譯器爲你做的。在網絡上有很多很好的例子,即使在SO上也是如此,向您展示了Fortran + OpenMP中並行do循環的常見超級結構。 –