承接前面两篇,这里直接逐一介绍和使用有关OpenMP的指令和函数

Directives

1、for

作用:for指令指定紧随其后的程序的循环的迭代必须由团队并行执行,只是假设已经建立了并行区域,否则它在单个处理器上串行执行。

格式:

 #pragma omp for [clause ...] newline
schedule (type [,chunk])
ordered
private (list)
firstprivate (list)
lastprivate (list)
shared (list)
reduction (operator:list)
collapse (n)
nowait for_loop

可以使用如下子句:

还可以通过Schedule子句(clause)设置for循环的并行化方法:(有关一种调度如何比其他调度更优化的讨论,请参阅http://openmp.org/forum/viewtopic.php?f=3&t=83

  • static:循环迭代被分成size chunk,然后静态的分配给各个线程,如果chunk没有被指定,则均匀地划分(如果可能)给各个线程
  • dynamic:循环迭代被分成size chunk,然后动态地分配给各个线程,当一个chunk完成时,被分配另外一个chunk。默认地chunk size为1
  • guided:当线程请求循环迭代时,迭代会动态地分配给块中地线程,直到没有剩余的块要被分配。与dynamic类似,不同的地方在于每次为线程分配chunk时都会变小,所以最初组中的循环体执行数目较大。初始大小与以下成正比,number_of_iterations / number_of_thread,后续块与之成比例,number_of_iterations_remaining / number_of_threads。
  • runtime: 循环的并行化方式不在编译时静态确定,而是推迟到程序执行时动态地根据环境变量OMP_SCHEDULE 来决定要使用的方法。此时在子句中指定chunk_size是非法的
  • auto:调度决策取决于编译器/运行时系统  

nowait子句:如果指定,则线程在循环结束时不同步

ordered子句:指定必须像在串行程序中一样执行循环的迭代,可以对for的部分使用

collapse子句:指定嵌套循环中应将多少循环折叠到一个大的迭代空间中,并根据schedule子句进行划分 。折叠迭代空间中的迭代顺序被确定为顺序执行它们。可以改善表现。

其它的子句后面会做介绍

限制:

  • 循环迭代变量必须是整数,并且所有线程的循环控制参数必须相同
  • 程序正确性不能取决于哪个线程执行特定迭代,需要确保程序的正确性
  • 从for指令关联的循环中分支是非法的
  • 必须将块大小指定为循环不变整数表达式,因为在不同线程的评估期间没有同步

示例

  #include <omp.h>
#define N 1000
#define CHUNKSIZE 100 main(int argc, char *argv[]) { int i, chunk;
float a[N], b[N], c[N]; /* Some initializations */
for (i=; i < N; i++)
a[i] = b[i] = i * 1.0;
chunk = CHUNKSIZE; #pragma omp parallel shared(a,b,c,chunk) private(i)
{ #pragma omp for schedule(dynamic,chunk) nowait
for (i=; i < N; i++)
c[i] = a[i] + b[i]; } /* end of parallel region */ }

2、section

作用:section是一种非迭代的工作共享结构,代码被划分成多个区域

格式:

 #pragma omp sections [clause ...]  newline
private (list)
firstprivate (list)
lastprivate (list)
reduction (operator: list)
nowait
{ #pragma omp section newline structured_block #pragma omp section newline structured_block }

注意:

  • 除非使用nowait子句,否则sections指令结尾都有一个隐含的障碍
  • 分区块里不能含有分支

示例

  #include <omp.h>
#define N 1000 main(int argc, char *argv[]) { int i;
float a[N], b[N], c[N], d[N]; /* Some initializations */
for (i=; i < N; i++) {
a[i] = i * 1.5;
b[i] = i + 22.35;
} #pragma omp parallel shared(a,b,c,d) private(i)
{ #pragma omp sections nowait
{ #pragma omp section
for (i=; i < N; i++)
c[i] = a[i] + b[i]; #pragma omp section
for (i=; i < N; i++)
d[i] = a[i] * b[i]; } /* end of sections */ } /* end of parallel region */ }

3、其它的不一一介绍了,请参阅:OpenMP

Clause

前面已经介绍了几个子句,这里主要介绍数据作用域子句。

1、private

作用:private子句将其列表中的变量声明为每个线程的私有变量

格式:

private (list)

要点:

  • 在组中的每个线程声明一个相同数据类型的变量
  • 所有对原始变量的引用全部替换为对新变量的引用
  • 被声明为private的变量应被认为未初始化

2、shared

作用:shared子句声明其列表中的变量,以便在团队中的所有线程之间共享

格式:

shared (list)

要点:

  • 共享变量仅存在于一个内存位置,并且所有线程都可以读取或写入该地址
  • 程序员有责任确保多个线程正确访问SHARED变量(例如通过CRITICAL部分)

3、reduction

作用:reduction子句对列表中的每个变量执行简化操作。为每个线程创建并初始化每个列表变量的私有副本。在缩减结束时,reduce变量应用于共享变量的所有私有副本,最终结果将写入全局共享变量。

格式:

 reduction (operator: list)

示例:

并行循环的迭代将以相同大小的块分配给团队中的每个线程(SCHEDULE STATIC);

在并行循环结构的末尾,所有线程将添加其“result”值以更新主线程的全局副本;

 #include <omp.h>

  main(int argc, char *argv[])  {

  int   i, n, chunk;
float a[], b[], result; /* Some initializations */
n = ;
chunk = ;
result = 0.0;
for (i=; i < n; i++) {
a[i] = i * 1.0;
b[i] = i * 2.0;
} #pragma omp parallel for \
default(shared) private(i) \
schedule(static,chunk) \
reduction(+:result) for (i=; i < n; i++)
result = result + (a[i] * b[i]); printf("Final result= %f\n",result); }

4、其它还有很多,省略

Run-time Library Routines

  • OpenMP API包含越来越多的运行时库例程
  • 对于C / C ++,所有运行时库例程都是实际的子例程。对于Fortran,有些实际上是函数,有些是子例程。
  • 对于C / C ++,通常需要包含 <omp.h>头文件

例如:

 #include <omp.h>
int omp_get_num_threads(void)

详细的函数介绍可见OpenMP入门教程(二)

Environment Variables

  • OpenMP提供一些环境变量来控制并行程序的执行
  • 所有的环境变量名都是大写字母,但是分配给它们的值不区分大小写

1、OMP_NUM_THREADS:设置在运行期间最大的线程数

setenv OMP_NUM_THREADS 

2、OMP_DYNAMIC:启用或禁用动态调整可用于执行并行区域的线程数。有效值为TRUE或FALSE

setenv OMP_DYNAMIC TRUE

3、OMP_PROC_BIND:启用或禁用绑定到处理器的线程。有效值为TRUE或FALSE。

setenv OMP_PROC_BIND TRUE

4、OMP_STACKSIZE:控制创建(非主)线程的堆栈大小

setenv OMP_STACKSIZE 2000500B
setenv OMP_STACKSIZE“ k”
setenv OMP_STACKSIZE 10M
setenv OMP_STACKSIZE“ M”
setenv OMP_STACKSIZE“ m”
setenv OMP_STACKSIZE“1G”
setenv OMP_STACKSIZE

4、还有很多其它的,省略

注:前面的运行API也能做与环境变量一样的工作,同时使用环境变量和运行时 API 会出现什么情况?运行时 API 将获得更高的优先权。

注:这是一个简单的OpenMP的练习网站:https://computing.llnl.gov/tutorials/openMP/exercise.html

参考链接:https://computing.llnl.gov/tutorials/openMP/#Abstract

OpenMP入门教程(三)的更多相关文章

  1. 无废话ExtJs 入门教程三[窗体:Window组件]

    无废话ExtJs 入门教程三[窗体:Window组件] extjs技术交流,欢迎加群(201926085) 1.代码如下: 1 <!DOCTYPE html PUBLIC "-//W3 ...

  2. PySide——Python图形化界面入门教程(三)

    PySide——Python图形化界面入门教程(三) ——使用内建新号和槽 ——Using Built-In Signals and Slots 上一个教程中,我们学习了如何创建和建立交互widget ...

  3. Elasticsearch入门教程(三):Elasticsearch索引&映射

    原文:Elasticsearch入门教程(三):Elasticsearch索引&映射 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文 ...

  4. RabbitMQ入门教程(三):Hello World

    原文:RabbitMQ入门教程(三):Hello World 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog. ...

  5. JasperReports入门教程(三):Paramters,Fields和Detail基本组件介绍

    JasperReports入门教程(三):Paramter,Field和Detail基本组件介绍 前言 前两篇博客带领大家进行了入门,做出了第一个例子.也解决了中文打印的问题.大家跟着例子也做出了de ...

  6. WebGL入门教程(三)-webgl动画

    前面文章: WebGL入门教程(一)-初识webgl WebGL入门教程(二)-webgl绘制三角形 WebGL动画有移动.旋转和缩放,我们将移动.旋转和缩放图形,然后将其绘制到屏幕上,称为变换(tr ...

  7. 无废话SharePoint入门教程三[创建网站集和网站]

    一.前言 前两篇文章讲解了什么是SharePoint,并且介绍了在SharePoint中一些常用的概念.但概念终究是概念,我们还是要脚踏实地的去动手实践.下面的文章对于了解SharePoint的人来说 ...

  8. Spring Cloud 入门教程(三): 配置自动刷新

    之前讲的配置管理, 只有在应用启动时会读取到GIT的内容, 之后只要应用不重启,GIT中文件的修改,应用无法感知, 即使重启Config Server也不行. 比如上一单元(Spring Cloud ...

  9. OpenMP 入门教程

    前两天(其实是几个月以前了)看到了代码中有 #pragma omp parallel for 一段,感觉好像是 OpenMP,以前看到并行化的东西都是直接躲开,既然躲不开了,不妨研究一下: OpenM ...

随机推荐

  1. oracle 建表 主键自增序列/////

    oracle 建表 主键自增序列 (2011-10-12 11:59:22) 转载▼ 标签: 杂谈 分类: oracle SQL> create table sms_activity(  2   ...

  2. 使用C#开发HTTP服务器系列之Hello World

    各位朋友大家好,我是秦元培,欢迎大家关注我的博客.从今天起,我将开始撰写一组关于HTTP服务器开发的系列文章.我为什么会有这样的想法呢?因为人们对Web技术存在误解,认为网站开发是Web技术的全部.其 ...

  3. idea把项目提交到svn服务器步骤

    1.设置忽略文件.文件夹 2.找到以下选项 3.选择要上传项目的svn服务器地方和svn管理目录 4.填完相关提交信息,点击share就ok啦

  4. 洛谷P1527 矩阵乘法——二维树状数组+整体二分

    题目:https://www.luogu.org/problemnew/show/P1527 整体二分,先把所有询问都存下来: 然后二分一个值,小于它的加到二维树状数组的前缀和里,判断一遍所有询问,就 ...

  5. saltstack源码-启动1

    决定看salt的源码了.干脆就从最基本的看起来,先看它的启动过程开始第一步用/etc/init.d/salt-master start 启动找到那个文件,发现有3种启动方式,suse,debian,c ...

  6. 腾讯API

    相关文档: API列表  腾讯开放平台联调工具集  公共返回码说明 SDK下载

  7. 09_传智播客iOS视频教程_自动释放池与NSLog函数

    不要管什么是自动释放池,现在给你讲你也听不懂.就业班才讲,不要知道太多,知道太多对你不好.电影里面死的最惨的人就是知道最多的人.把代码写到自动释放池里面就可以了.NSLog是printf的增强版,它增 ...

  8. Android中string.xml中的的标签xliff:g(转载)

    转自:http://blog.csdn.net/xuewater/article/details/25687987 在资源文件中写字符串时,如果这个字符串时动态的,又不确定的值在里面,我们就可以用xl ...

  9. AOP日志框架实现

    AOP日志框架实现 JDK动态代理实现日志框架 首先,在项目包com.ay.test 下创建业务接口类BusinessClassService,具体代码如下: BusinessC lassServic ...

  10. Python——nose实例示例

    1. 脚本示例: 执行结果: 2. 脚本示例: 执行结果: 由以上截图可以看出执行顺序: 大体可以得出如下结论: 1)测试的顺序总体上按照包—>模块—>类的顺序进行: 2) 然后按照测试用 ...