免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
123下一页
最近访问板块 发新帖
查看: 4353 | 回复: 20
打印 上一主题 下一主题

[C] 一个看似简单,但包含技术难点的线程问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-10-18 10:33 |只看该作者 |倒序浏览
我最近在使用OPENMP在各个环境下做时间,其中在WINDOWS的vs2008以及linux环境下都很顺利,但是在AIX环境下,CPU使用率总是提不上去,后来通过修改pthread_attr_setscope设置在SYSTEM中竞争,略微好点,但是效果仍然不明显。准备转向自己动手做线程试试,不过原先做线程处理都是任务型的,对于算法的并行没有做过,也就是说想用线程对循环计算进行并行化,我的数据是一个8192 * 8000的short数据,而且还分7层,所以是个数组,一次数组下标是很重要的,我就做了一个例子来确定线程计数器的唯一性(主要怕出现计数下标不唯一,造成的计算结果错误),我的实验程序如下:

  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <strings.h>
  4. #include <stdlib.h>
  5. #include <pthread.h>

  6. int count = 1000000;
  7. int k = 0;
  8. pthread_mutex_t mutex;

  9. void *thr_fn(void *arg)
  10. {
  11.         int tmp;
  12. pthread_detach(pthread_self());

  13. pthread_mutex_lock(&mutex);
  14.    k++;
  15.    tmp = k;
  16. pthread_mutex_unlock(&mutex);

  17.    int kk = 0;
  18.    for(int z = 0; z < count; z++)
  19.    {
  20.       kk = z * z /kk + kk;
  21.    }

  22.    printf("%05d end pid:%u  tid:%u\n", tmp, getpid(), pthread_self());
  23.    pthread_exit(0);
  24.    return NULL;
  25. }

  26. int main()
  27. {
  28.    int err;
  29.    int i = 0;
  30.    pthread_t tid;
  31.    for(; i < 10000; i++)
  32.    {
  33. printf("i = [%d]\n", i);
  34.    err = pthread_create(&tid,NULL,thr_fn, NULL);
  35.    if(err=0)
  36.       printf("can't create thread:%s\n",strerror(err));
  37.    }
  38.    sleep(10);
  39.    exit(0);
  40. }
复制代码
内部使用线程锁作为保护机制,但是结果发现仍然有8个点被击穿了
num = [227]
num = [232]
num = [2770]
num = [5065]
num = [5242]
num = [7514]
num = [8446]
num = [9487]
有重复,难道线程内部不能使用流水呀,计数值么?

另外,main中的i是不能作为计数值参数传递的,以为竞争更厉害。

论坛徽章:
44
15-16赛季CBA联赛之浙江
日期:2021-10-11 02:03:59程序设计版块每日发帖之星
日期:2016-07-02 06:20:0015-16赛季CBA联赛之新疆
日期:2016-04-25 10:55:452016科比退役纪念章
日期:2016-04-23 00:51:2315-16赛季CBA联赛之山东
日期:2016-04-17 12:00:2815-16赛季CBA联赛之福建
日期:2016-04-12 15:21:2915-16赛季CBA联赛之辽宁
日期:2016-03-24 21:38:2715-16赛季CBA联赛之福建
日期:2016-03-18 12:13:4015-16赛季CBA联赛之佛山
日期:2016-02-05 00:55:2015-16赛季CBA联赛之佛山
日期:2016-02-04 21:11:3615-16赛季CBA联赛之天津
日期:2016-11-02 00:33:1215-16赛季CBA联赛之浙江
日期:2017-01-13 01:31:49
2 [报告]
发表于 2012-10-18 10:58 |只看该作者
  1. int kk = 0;
  2. ...
  3.       kk = z * z /kk + kk;
复制代码
你确定这程序能正常跑?

论坛徽章:
0
3 [报告]
发表于 2012-10-18 10:59 |只看该作者
大家看的人好多,有人知道么?其实这个例子前我还做了一些其他的实验,例如在main的for中使用临时变量,将i值传递给临时变量传递给线程;或者使用malloc分配变量空间啦,结果发现都不成,尤其是malloc方法,竟然使用的是同一地址,如果不free原先用过地址是不同地址,结果也是正确的,但是意味着内存溢出问题;

论坛徽章:
0
4 [报告]
发表于 2012-10-18 11:00 |只看该作者
可以,kk = z * z /kk + kk;只是我随便编的一个算法,怕被编译器给优化了,所以使用了kk在内的计算。其实换一个公式对实验并不冲突。

论坛徽章:
3
CU大牛徽章
日期:2013-03-13 15:15:08CU大牛徽章
日期:2013-03-13 15:26:06CU大牛徽章
日期:2013-03-13 15:26:47
5 [报告]
发表于 2012-10-18 11:09 |只看该作者
同2楼所问,另外mutex不用初始化的么?

论坛徽章:
0
6 [报告]
发表于 2012-10-18 11:13 |只看该作者
其实,我的实验是不关心kk = z * z /kk + kk;的结果的,主要是为了验证tmp的值唯一在每个线程中;所以kk = z * z /kk + kk;用什么公式都可以;另外我是一个简单程序mutex开始执行,默认值就是0;

论坛徽章:
0
7 [报告]
发表于 2012-10-18 12:03 |只看该作者
首先,请查看关键字 volatile
第二,这种数值可以用
win:Interlocked系列API
linux gcc:__sync_add_and_fetch 那一批函数
mac : OSAtomicIncrement32Barrier 一系列

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
8 [报告]
发表于 2012-10-18 12:33 |只看该作者
mutex写错,kk等于没有。

楼主放一段正确的程序上来再说,不然没法求证。

论坛徽章:
0
9 [报告]
发表于 2012-10-18 12:38 |只看该作者
楼上说的有道理,我参考资料http://blog.chinaunix.net/uid-14735472-id-111027.html,修改了一下代码:
  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <strings.h>
  4. #include <stdlib.h>
  5. #include <pthread.h>

  6. int count = 1000000;
  7. volatile int k = 0;
  8. pthread_mutex_t mutex;

  9. void *thr_fn(void *arg)
  10. {
  11.         volatile int tmp;
  12. pthread_detach(pthread_self());

  13. pthread_mutex_lock(&mutex);
  14.    k++;
  15.    tmp = k;
  16. pthread_mutex_unlock(&mutex);

  17.    int kk = 0;
  18.    for(int z = 0; z < count; z++)
  19.    {
  20.       kk = z * z /kk + kk;
  21.    }

  22.    printf("%05d end pid:%u  tid:%u\n", tmp, getpid(), pthread_self());
  23.    pthread_exit(0);
  24.    return NULL;
  25. }

  26. int main()
  27. {
  28.    int err;
  29.    int i = 0;
  30.    pthread_t tid;
  31.    for(; i < 10000; i++)
  32.    {
  33. printf("i = [%d]\n", i);
  34.    err = pthread_create(&tid,NULL,thr_fn, NULL);
  35.    if(err=0)
  36.       printf("can't create thread:%s\n",strerror(err));
  37.    }
  38.    sleep(10);
  39.    exit(0);
  40. }
复制代码
单执行结果任然有:5个重复点
num = [272]
num = [272]
num = [2375]
num = [7921]
num = [8362]

论坛徽章:
0
10 [报告]
发表于 2012-10-18 12:41 |只看该作者
我已经说了其实,kk那段不重要,这个实验时为了验证tmp值是否是唯一的,即从0到9999值,中间是否有重复值。楼上说的volatile很有道理,但是修改后仍然有重复值,真奇怪。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP