矩阵乘法的OpenMP实现与性能分析报告文档格式.docx

上传人:b****3 文档编号:6729932 上传时间:2023-05-07 格式:DOCX 页数:12 大小:116.99KB
下载 相关 举报
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第1页
第1页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第2页
第2页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第3页
第3页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第4页
第4页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第5页
第5页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第6页
第6页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第7页
第7页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第8页
第8页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第9页
第9页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第10页
第10页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第11页
第11页 / 共12页
矩阵乘法的OpenMP实现与性能分析报告文档格式.docx_第12页
第12页 / 共12页
亲,该文档总共12页,全部预览完了,如果喜欢就下载吧!
下载资源
资源描述

矩阵乘法的OpenMP实现与性能分析报告文档格式.docx

《矩阵乘法的OpenMP实现与性能分析报告文档格式.docx》由会员分享,可在线阅读,更多相关《矩阵乘法的OpenMP实现与性能分析报告文档格式.docx(12页珍藏版)》请在冰点文库上搜索。

矩阵乘法的OpenMP实现与性能分析报告文档格式.docx

sys/time.h>

timevalstart,end;

gettimeofday(&

start,NULL);

end,NULL);

cout<

<

"

executiontime:

(end.tv_sec-start.tv_sec)+(double)(end.tv_usec-start.tv_usec)/1000000<

seconds"

<

endl;

答:

在windows下使用MicrosofeVisualStudio编程,源代码如下:

omp.h>

stdio.h>

#defineNN2000

inta[NN][NN],b[NN][NN];

longlongc[NN][NN];

voidsolve(intn,intnum_thread)

{

inti,j,t,k,time;

clock_tstartTime,endTime;

longlongsum;

omp_set_num_threads(num_thread);

for(i=0;

i<

n;

i++)//对矩阵a和矩阵b进行初始化

{

t=i+1;

for(j=0;

j<

j++)

{

a[i][j]=t++;

b[i][j]=1;

}

}

startTime=clock();

sum=0;

#pragmaompparallelshared(a,b,c)private(i,j,k)

#pragmaompforschedule(dynamic)

i++)

c[i][j]=0;

for(k=0;

k<

k++)

{

c[i][j]+=a[i][k]*b[k][j];

}

}

i++)for(j=0;

j++)sum+=c[i][j];

endTime=clock();

time=endTime-startTime;

printf("

sum=%lldtime=%dms\n"

sum,time);

intmain()

intn,num_thread;

while(scanf("

%d%d"

&

n,&

num_thread)!

=EOF)

solve(n,num_thread);

return0;

 

2.分析矩阵相乘程序的执行时间、加速比和效率:

方阵阶固定为1000,节点数分别取1、2、4、8、16和32时,为减少误差,每项实验进行5次,取平均值作为实验结果。

串行执行时程序的执行时间为:

T=15.062s

加速比=顺序执行时间/并行执行时间

效率=加速比/节点数

表1不同节点数下程序的执行时间(秒)

节点数

实验结果

1

2

4

8

16

32

第1次

16.640

8.172

4.078

2.125

1.093

0.594

第2次

16.422

8.156

4.172

2.141

1.078

0.578

第3次

16.406

8.266

1.094

0.563

第4次

16.781

4.079

2.109

第5次

8.171

平均值

16.5342

8.1874

4.0970

2.1250

1.0904

0.5752

图1不同节点数下程序的执行时间

图2不同节点数下程序的加速比

图3不同节点数下程序的效率

执行时间的分析:

随着节点数的增加,程序的执行时间减少,大概可以从结果中得出,随着节点书的增加一倍,执行时间减少一半

加速比的分析:

随着节点数的增加,程序的加速比增加,大概可以从结果中得出,随着节点书的增加一倍,加速相应的增加接近一倍

效率的分析:

随着节点数的增加,程序的效率逐渐减少

3.分析矩阵相乘程序的问题规模与效率的关系:

固定节点数为4,让方阵阶从200到1600之间变化,每隔100取一个值。

(为了减少时间,每项实验可只执行1次)

表2相同节点数下不同问题规模程序的执行时间与效率

方阵阶数

并行执

行时间

串行执

效率

200

0.015

0.047

0.783333

300

0.016

0.109

1.703125

400

0.063

0.297

1.178571

500

0.156

0.657

1.052885

600

0.406

1.64

1.009852

700

0.907

3.578

0.986218

800

1.609

6.36

0.988191

900

2.578

10.109

0.980314

1000

3.812

14.891

0.976587

1100

5.39

21.032

0.97551

1200

7.344

28.734

0.978145

1300

9.688

37.937

0.978969

1400

12.422

48.64

0.978908

1500

15.656

60.938

0.973077

1600

19.234

74.829

0.972614

图3.1不同问题规模下程序的效率

问题规模与效率的关系分析:

随着问题规模的增加,程序的效率趋于稳定,但是略微有点下降。

嵌套循环中,如果外层循环迭代次数较少时,如果将来CPU核数增加到一定程度时,创建的线程数将可能小于CPU核数。

另外如果内层循环存在负载平衡的情况下,很难调度外层循环使之达到负载平衡。

下面以矩阵乘法作为例子来讲述如何将嵌套循环并行化,以满足上述扩展性和负载平衡需求。

一个串行的矩阵乘法的函数代码如下:

/**矩阵串行乘法函数

@paramint*a-指向要相乘的第个矩阵的指针

@paramintrow_a-矩阵a的行数

@paramintcol_a-矩阵a的列数

@paramint*b–指向要想成的第个矩阵的指针

@paramintrow_b-矩阵b的行数

@paramintcol_b-矩阵b的列数

@paramint*c-计算结果的矩阵的指针

@paramintc_size-矩阵c的空间大小(总元素个数)

@returnvoid–无

*/

voidMartrix_Multiply(int*a,introw_a,intcol_a,

int*b,introw_b,intcol_b,

int*c,intc_size)

If(col_a!

=row_b||c_size<

row_a*col_b)

return;

inti,j,k;

//#pragmaompforprivate(i,j,k)

for(i=0;

row_a;

introw_i=i*col_a;

introw_c=i*col_b;

for(j=0;

col_b;

c[row_c+j]=0;

for(k=0;

row_b;

c[row_c+j]+=a[row_i+k]*b[k*col_b+j];

如果在外层循环前面加上OpenMP的for语句时,它就变成了一个并行的矩阵乘法函数,但是这样简单地将其并行化显然无法满足前面所述的扩展性需求。

其实可以采用一个简单地方法将最外层循环和第2层循环合并成一个循环,下面便是采用合并循环后的并行实现。

voidParallel_Matrix_Multiply(int*a,introw_a,intcol_a,

int*b,introw_b,intcol_b,

int*c,intc_size)

=row_b)

intindex;

intborder=row_a*col_b;

i=0;

j=0;

//#pragmaompparallelprivate(i,j,k)num_threads(dtn(border,1))

for(index=0;

index<

border;

index++)

i=index/col_b;

j=index%col_b;

从上面代码可以看出,合并后的循环便捷border=row_a*col_b;

即等于原来的两个循环边界之积,然后再循环中计算出原来的外层循环和第2层循环的迭代变量i和j,采用除法和取余来求出i和j的值。

需要值得注意的是,上面求i和j的值必须要保证循环迭代的独立性,即不能有循环迭代间的依赖关系。

不能讲求i和j的值得过程优化成如下的形式

if(j==col_b)

j=0;

i++;

//.......此处代表实际的矩阵乘法代码

j++;

上面这种优化,省去了除法,效率高,但是只能在串行代码中使用,因为它存在循环迭代间的依赖关系,无法将其正确地并行

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 法律文书 > 调解书

copyright@ 2008-2023 冰点文库 网站版权所有

经营许可证编号:鄂ICP备19020893号-2