嗨,写了一个非常简单的例子,说明如何使用 omp flush 交换数据,以生产者-> 消费者的方式,在线程中我发现了一个有趣的行为。
int a=-1;
int flag=1;
int count=0;
#pragma omp parallel num_threads(2)
{
int TID;
TID=omp_get_thread_num();
#pragma omp sections
{
#pragma omp section /////////// Producer
{
for(int i=0; i<9;i++)
{
a=i;
#pragma omp flush(a)
flag=1;
printf("Producer a: %d flag:%d TID %d \n",a,flag,TID);
while(flag)
{
#pragma omp flush(flag)
}
}
flag=2;
#pragma omp flush(flag)
} // end producer
#pragma omp section /////////// Consumer
{
while(1) {
count++;
flag=0;
while(!flag)
{
#pragma omp flush(flag)
}
#pragma omp flush(a)
printf("Consumer a: %d Flag: %d count %d TID %d \n",a,flag,count,TID);
if (flag==2) break; // no more data
} // end while(1)
}// end consumer
}// end sections
使用这个非常简单的代码会产生错误的输出:
生产者 a:0 标志:1 TID 0
生产者 a:1 标志:1 TID 0
消费者 a:1 标志:1 个计数 1 TID 1
生产者 a:2 标志:1 TID 0
消费者 a:2 标志:1 个计数 2 TID 1
生产者 a:3 标志:1 TID 0
消费者 a:3 标志:1 个计数 3 TID 1
生产者 a:4 标志:1 TID 0
消费者 a:4 标志:1 个计数 4 TID 1
生产者 a:5 标志:1 TID 0
消费者 a:5 标志:1 个计数 5 TID 1
生产者 a:6 标志:1 TID 0
消费者 a:6 标志:1 个计数 6 TID 1
生产者 a:7 标志:1 TID 0
消费者 a:7 标志:1 个计数 7 TID 1
生产者 a:8 标志:1 TID 0
消费者 a:8 标志:1 个计数 8 TID 1
消费者 a:8 标志:2 计数 9 TID 1
错误是生成的第一个数据 a=0 被消费者忽略。
如果我只是颠倒部分的顺序,让生产者成为线程 1,那么一切都很好.....
生产者 a:0 标志:1 TID 1
消费者 a:0 标志:1 计数 1 TID 0
生产者 a:1 标志:1 TID 1
消费者 a:1 标志:1 计数 2 TID 0
....
我的错误是什么?
.....
在与 Ejd(感谢)进行有趣的讨论后,代码被编辑为:
int a=-1;
int flag=0;
int count=0;
#pragma omp parallel num_threads(2)
{
int TID;
TID=omp_get_thread_num();
#pragma omp sections
{
#pragma omp section /////////// Consumer
{
while(1) {
count++;
if (flag) printf("Consumer a: %d Flag: %d count %d TID %d \n",a,flag,count,TID);
flag=0;
while(!flag)
{
#pragma omp flush(flag)
}
if (flag==2) break; // no more data
} // end while(1)
}// end consumer
#pragma omp section /////////// Producer
{
for(int i=0; i<9;i++)
{
a=i;
printf("Producer a: %d flag:%d TID %d \n",a,flag,TID);
flag=1;
while(flag)
{
#pragma omp flush(flag,a)
}
}
flag=2;
#pragma omp flush(flag)
} // end producer
}// end sections
现在效果很好。谢谢 !
最佳答案
不幸的是,使用flush 比乍一看要复杂得多。即使是 OpenMP 专家也很难正确使用它。部分问题在于,与列表齐平的定义不正确。基本上它是允许移动的,所以如果你有一个形式的序列:
a = ...
#pragma omp flush(a)
b = ...
#pragma omp flush(b)
flush(a) 必须在 a 的设置之后,但可以在 set 和 flush(b) 之后移动。它只需要在下一次使用 a 之前发生。
在任何情况下,做你想做的最好的方法是使用没有 列表的 flush 并在你感兴趣的每组变量之后执行刷新,并在读取你感兴趣的每个变量之前执行刷新在。
另一个问题是您没有正确交接。您不能在消费者中为生产者设置标志以生成另一个数字,直到消费者实际消费了产生的值。
关于c - #pragma omp flush 使线程间交换数据,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/5110816/