基于OpenMp的并行编程
功能:并行处理比较耗时的for循环
在OpenMP中,对for循环并行化的任务调度使用schedule子句来实现:
- 使用格式:schedule(type[,size])
- type参数表示调度类型:static、dynamic、guided
- size参数(可选):分配给每个线程的连续迭代计算的次数
- 静态调度(static)
大部分编译器在没有使用schedule子句的时候,默认是static调度。static在编译的时候就已经确定了,那些循环由哪些线程执行。
当不使用size 时,将给每个线程分配┌N/t┐个迭代。当使用size时,将每次给线程分配size次迭代。
#pragma omp parallel for schedule(static)
//#pragma omp parallel for schedule(static,2)
for (int i = 0; i < 10; ++i)
{
printf("i=%d, thread_id=%d\n", i, omp_get_thread_num());
}
- 动态调度(dynamic)
动态地将迭代分配到各个线程,动态调度可以使用size参数也可以不使用size参数,不使用size参数时是将迭代逐个地分配到各个线程,使用size参数时,每次分配给线程的迭代次数为指定的size次
#pragma omp parallel for schedule(dynamic)
//#pragma omp parallel for schedule(dynamic,2)
for (int i = 0; i < 10; ++i)
{
printf("i=%d, thread_id=%d\n", i, omp_get_thread_num());
}
从结果看出:这个调度是逐个将任务分配到每一个核心,然后哪个执行完了就接着分配。如果指定size为2,就会每一次为每一个核心连续分配两个任务
- guided调度(guided)
guided调度是一种采用指导性的启发式自调度方法。开始时每个线程会分配到较大的迭代块,之后分配到的迭代块会逐渐递减。迭代块的大小会按指数级下降到指定的size大小,如果没有指定size参数,那么迭代块大小最小会降到1
#pragma omp parallel for schedule(guided)
for (int i = 0; i < 10; ++i)
{
printf("i=%d, thread_id=%d\n", i, omp_get_thread_num());
}
- 三种运行方式总结:
静态调度static:每次哪些循环由那个线程执行时固定的,编译调试。由于每个线程的任务是固定的,但是可能有的循环任务执行快,有的慢,不能达到最优。
动态调度dynamic:根据线程的执行快慢,已经完成任务的线程会自动请求新的任务或者任务块,每次领取的任务块是固定的。
启发式调度guided:每个任务分配的任务是先大后小,指数下降。当有大量任务需要循环时,刚开始为线程分配大量任务,最后任务不多时,给每个线程少量任务,可以达到线程任务均衡。
open MP 与串行执行 对比:
四种调度方式实例: dynamic、guided、runtime、static