并行实现快速排序

1:环境:Linux     系统CentOS 7

2.原理:

OpenMP基本概念

OpenMP是一种用于共享内存并行系统的多线程程序设计方案,支持的编程语言包括C、C++和Fortran。OpenMP提供了对并行算法的高层抽象描述,特别适合在多核CPU机器上的并行程序设计。编译器根据程序中添加的pragma指令,自动将程序并行处理,使用OpenMP降低了并行编程的难度和复杂度。当编译器不支持OpenMP时,程序会退化成普通(串行)程序。程序中已有的OpenMP指令不会影响程序的正常编译运行。

在VS中启用OpenMP很简单,很多主流的编译环境都内置了OpenMP。在项目上右键->属性->配置属性->C/C++->语言->OpenMP支持,选择“是”即可。

OpenMP执行模式

OpenMP采用fork-join的执行模式。开始的时候只存在一个主线程,当需要进行并行计算的时候,派生出若干个分支线程来执行并行任务。当并行代码执行完成之后,分支线程会合,并把控制流程交给单独的主线程。

一个典型的fork-join执行模型的示意图如下:

OpenMP编程模型以线程为基础,通过编译制导指令制导并行化,有三种编程要素可以实现并行化控制,他们分别是编译制导、API函数集和环境变量。

编译制导

编译制导指令以#pragma omp 开始,后边跟具体的功能指令,格式如:#pragma omp 指令[子句[,子句] …]。常用的功能指令如下:

  • parallel:用在一个结构块之前,表示这段代码将被多个线程并行执行;
    for:用于for循环语句之前,表示将循环计算任务分配到多个线程中并行执行,以实现任务分担,必须由编程人员自己保证每次循环之间无数据相关性;
    parallel for:parallel和for指令的结合,也是用在for循环语句之前,表示for循环体的代码将被多个线程并行执行,它同时具有并行域的产生和任务分担两个功能;
    sections:用在可被并行执行的代码段之前,用于实现多个结构块语句的任务分担,可并行执行的代码段各自用section指令标出(注意区分sections和section);
    parallel sections:parallel和sections两个语句的结合,类似于parallel for;
    single:用在并行域内,表示一段只被单个线程执行的代码;
    critical:用在一段代码临界区之前,保证每次只有一个OpenMP线程进入;
    flush:保证各个OpenMP线程的数据影像的一致性;
    barrier:用于并行域内代码的线程同步,线程执行到barrier时要停下等待,直到所有线程都执行到barrier时才继续往下执行;
    atomic:用于指定一个数据操作需要原子性地完成;
    master:用于指定一段代码由主线程执行;
    threadprivate:用于指定一个或多个变量是线程专用,后面会解释线程专有和私有的区别。

相应的OpenMP子句为: 


  • private:指定一个或多个变量在每个线程中都有它自己的私有副本;
    firstprivate:指定一个或多个变量在每个线程都有它自己的私有副本,并且私有变量要在进入并行域或任务分担域时,继承主线程中的同名变量的值作为初值;
    lastprivate:是用来指定将线程中的一个或多个私有变量的值在并行处理结束后复制到主线程中的同名变量中,负责拷贝的线程是for或sections任务分担中的最后一个线程; 
    reduction:用来指定一个或多个变量是私有的,并且在并行处理结束后这些变量要执行指定的归约运算,并将结果返回给主线程同名变量;
    nowait:指出并发线程可以忽略其他制导指令暗含的路障同步;
    num_threads:指定并行域内的线程的数目; 
    schedule:指定for任务分担中的任务分配调度类型;
    shared:指定一个或多个变量为多个线程间的共享变量;
    ordered:用来指定for任务分担域内指定代码段需要按照串行循环次序执行;
    copyprivate:配合single指令,将指定线程的专有变量广播到并行域内其他线程的同名变量中;
    copyin:用来指定一个threadprivate类型的变量需要用主线程同名变量进行初始化;
    default:用来指定并行域内的变量的使用方式,缺省是shared。

利用omp_set_num_threads()来设置线程数,

利用#pragma omp parallel sections 声明下面大括号中的语句要并行多线程执行;

利用#pragma omp section 分配线程。

看懂下列helloworld的代码对openmp并行就会有一定了解。

 

代码实现:

#include<stdio.h>
#include<omp.h>
#include<time.h>
#include<stdlib.h>

int Partition(int* A,int low,int hight)
{
int pivot=A[low];
while(low<hight)
{
while(low<hight&&A[hight]>=pivot)--hight;
A[low]=A[hight];
while(low<hight&&A[low]<=pivot)++low;
A[hight]=A[low];
}
A[low]=pivot;

return low;

}

void QuickSort(int* A,int low,int hight)
{

if(low<hight)
{
int pivotpos=Partition(A,low,hight);
#pragma omp parallel sections
{
#pragma omp section
QuickSort(A,low,pivotpos-1);
#pragma omp section
QuickSort(A,pivotpos+1,hight);
}
}

}


int main(int argc,char* argv[])
{
int n=atoi(argv[2]);
int size=atoi(argv[1]);
int * A=(int*)malloc(sizeof(int) * size);

int* B=(int*)malloc(sizeof(int)*size);
srand(time(NULL)+rand());
for(int i=0;i<size;i++)
{
A[i]=rand();
B[i]=A[i];

}
double starttime=omp_get_wtime();
omp_set_num_threads(n);


QuickSort(A,0,size-1);
double endtime=omp_get_wtime();
for(int i=0;i<=10&&i<size;i++)
{
printf("%d  ",A[i]);
}
printf("\nbingxingtime:%lfs\n",endtime-starttime);

double starttime1=omp_get_wtime();
omp_set_num_threads(1);
QuickSort(B,0,size-1);
double endtime1=omp_get_wtime();
for(int i=0;i<=10&&i<size;i++)
{

printf("%d\n",B[i]);
} 
printf("\nchunxing\n:%lfs\n",endtime1-starttime1);




return 0;
}







编译:g++ -fopenmp quirtsort.cpp -o quirtsort

运行:./quirtsort 10000 4     

注意第一个参数为快排数据的大小,第二个参数为线程数,例如我这里就设置了四个线程。

结果如下:

 可以看到并行和串行的时间差不多,那是因为并行的通询时间话的比较多,串行不需要通询时间

当数据量大到百万及时并行明显快于串行。

如下:

 多数时候是串行快:

我查了一下原因如下:

并行接口速度比串行接口速度快,这是若干年前的情况了。 在实际时钟频率比较低的情况下,并口因为可以同时传输若干比特,速率确实比串口快。 但是,随着技术的发展,时钟频率越来越高,并行导线之间的相互干扰越来越严重。 并行接口因为有多条并行且紧密的导线,但时钟频率提高的一定程度时,传输的数据已经无法恢复。 而串口因为导线少,线间干扰容易控制,反而可以通过不断提高时钟频率来提高传输速率。

  • 2
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值