UNIX环境高级编程——pthread_create的问题

     linux 下常用的创建多线程函数pthread_create(pthread_t * thread , pthread_attr_t * attr , void *(*start_routine)(void*) , void *args);

其中第一个参数用来保存线程信息,第二个参数指新线程的运行属性,可以设置为NULL,第三个参数为自定义的线程函数,第四个参数就是线程函数需要用到的参数,一般如果要传递多个参数,可以设置为结构体(struct)类型,这里我们使用int类型的变量。  

     下面我着重讨论一个用for结构来创建多个线程时参数传递的问题:

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>#define th_pop 20 pthread_mutex_t mutex;pthread_t a_thread[th_pop];void * thread_func(void *args)
{pthread_mutex_lock(&mutex);int t_id = *(int*)args;printf("the id of this thread is %d\n",t_id);pthread_mutex_unlock(&mutex);    return (void*)NULL;
}void init()
{pthread_mutex_init(&mutex, NULL);int i;for( i=0; i<th_pop; i++){pthread_create(&a_thread[i] , NULL , thread_func , &i);}//wait the end of the threads;for( i=0; i<th_pop; i++){int res = pthread_join(a_thread[i] , NULL);if(res != 0)printf("the thread id: %d ends fail \n",i);}pthread_mutex_destroy(&mutex);}int main()
{init();return 0;
}

运行结果:

huangcheng@ubuntu:~$ ./a.out
the id of this thread is 2
the id of this thread is 8
the id of this thread is 9
the id of this thread is 9
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20
the id of this thread is 20


看到这个结果有没有感觉到有什么不对呢?可能你会感觉到很纳闷,怎么出现了那么多的id=0的结果呢?其实这个认真分析一下并不难理解:

首先pthread_create函数传递的是一个指针型的参数,即传递的是一个地址而已,这样在执行for结构时:

for(int i=0; i<th_pop; i++)
{pthread_create(&a_thread[i] , NULL , thread_func , &i);
}


     该for循环快速执行完成,并且将i置为20,故而传递的地址指向的内容为20,同时其它的线程还没来得及执行:
 int t_id = *(int*)args;

这样就使得多个线程都指向同一个地址,内容为20,解决该问题的一个办法为中for结构中加入sleep(1),这样当sleep时间大于线程函数执行时间,就可以得到一个正确的结果,不过这种办法剥掉了并发性,并不可取,下面我们采用另一种方法。

我们只修改init()函数

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>#define th_pop 20 //pthread_mutex_t mutex;pthread_t a_thread[th_pop];void * thread_func(void *args)
{pthread_mutex_lock(&mutex);int t_id = *(int*)args;printf("the id of this thread is %d\n",t_id);pthread_mutex_unlock(&mutex);    return (void*)NULL;
}void init()
{pthread_mutex_init(&mutex, NULL);int thread_id[th_pop];int i;for( i=0; i<th_pop; i++)thread_id[i] = i;for( i=0; i<th_pop; i++){int *t = thread_id +i;pthread_create(&a_thread[i] , NULL , thread_func , (void*)t);}//wait the end of the threads;for( i=0; i<th_pop; i++){int res = pthread_join(a_thread[i] , NULL);if(res != 0)printf("the thread id: %d ends fail \n",i);}pthread_mutex_destroy(&mutex);}int main()
{init();return 0;
}
运行结果:

huangcheng@ubuntu:~$ ./a.out
the id of this thread is 3
the id of this thread is 2
the id of this thread is 1
the id of this thread is 4
the id of this thread is 5
the id of this thread is 6
the id of this thread is 7
the id of this thread is 8
the id of this thread is 9
the id of this thread is 10
the id of this thread is 11
the id of this thread is 12
the id of this thread is 13
the id of this thread is 14
the id of this thread is 15
the id of this thread is 16
the id of this thread is 17
the id of this thread is 18
the id of this thread is 19
the id of this thread is 0

从这个例子中我们应该明白,要避免直接在传递的参数中传递发生改变的量,否则会导致结果不可测。

解决这类问题的办法:

(1)复制一份到堆里面。
(2)加锁。(一般做法)
(3)用结构体数组。(推荐这个)
(4)sleep。(不推荐)


转载于:https://www.cnblogs.com/wangfengju/p/6172684.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/575871.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

redis value最大值_Redis 的 maxmemory 和 dbnum 默认值都是多少?对于最大值会有限制吗?...

一、Redis 的默认配置了解 Redis 的都知道&#xff0c;Redis 服务器状态有很多可配置的默认值。例如&#xff1a;数据库数量&#xff0c;最大可用内存&#xff0c;AOF 持久化相关配置和 RDB 持久化相关配置等等。我相信&#xff0c;关于 AOF 持久化和 RDB 持久化的配置大家都很…

java成长之路

在Java语言中&#xff0c;一维数组的使用类似于变量的使用&#xff0c;分为以下4个步骤&#xff1a; 声明数组 创建数组 初始化数组 引用数组元素 语法&#xff1a;数据类型 [] 数组名 ; 或数据类型 数组名 []; int[] age; // 用于存储年龄 String names[]; // 用于存储…

程序员》推荐C++ 图书三人谈

程序员》推荐C 图书三人谈 主持人&#xff1a;熊节&#xff08;透明&#xff09;&#xff0c;《程序员》杂志编辑&#xff0c;C-View成员 嘉 宾&#xff1a;孟岩&#xff08;梦魇&#xff09;,联想公司掌上设备事业部应用开发处任职&#xff0c;C-View成员。与侯捷先生合译《C …

谁是对象谁是类?你搞清楚了吗?

今天带大家了解一下Java中类与对象的知识&#xff0c;一起来学习吧 现实世界中的所有具体的事物都可以视为对象 Java中万物皆对象&#xff0c;属性——对象具有的各种特征 每个对象的每个属性都拥有特定值&#xff0c;类是对对象的描述&#xff0c;是创建对象的“模板”&am…

实现页面适配_微信公众号文章页面适配深色模式

最近安卓微信7.0.10正式版发布&#xff0c;更新过后&#xff0c;很多用户发现&#xff0c;之前在测试版中对系统深色模式的适配功能被取消了&#xff0c;小伙伴们对此很是不满&#xff0c;好在Android 10系统手机用户占比很少&#xff0c;影响范围还不是很大&#xff0c;并且也…

推荐系统相关资源搜集

1.open source code: https://mymediaproject.codeplex.com/ 2.推荐系统常见推荐算法比较&#xff1a; http://fantasticinblur.iteye.com/blog/1333899 3.Hulu项亮的博士论文以及论著&#xff1a; http://vdisk.weibo.com/s/KCCWk 4.MovieLens的Dataset: http://www.grouplens…

iframe悬浮在html上_HTML Iframe

HTML Iframeiframe 用于在网页内显示网页。添加 iframe 的语法URL 指向隔离页面的位置。Iframe - 设置高度和宽度height 和 width 属性用于规定 iframe 的高度和宽度。属性值的默认单位是像素&#xff0c;但也可以用百分比来设定(比如 "80%")。实例&#xff1a;html&…

cdn需要备案吗_cdn需要备案么

cdn需要备案么&#xff1f;需要的&#xff0c;使用国内的cdn节点&#xff0c;必须要有备案才可以做cdn&#xff0c;相关部门为防止cdn滥用&#xff0c;要求必须先备案。国内节点只要是大型数据中心的&#xff0c;基本都要备案。使用CDN产品是需要备案的&#xff0c;但不限制在提…

POJ 3667 Hotel

题目大意&#xff1a; 1 a:询问是不是有连续长度为a的空房间,有的话住进最左边 2 a b:将[a,ab-1]的房间清空 思路&#xff1a;线段树的区间合并。 用cov记录区段的状态&#xff0c;-1代表没有被更新&#xff0c;0代表空闲&#xff0c;1代表是有人入住的。 用lmax代表从左端点开…

Mysql能删了重装吗_mysql卸载重新安装

1.控制面板——》所有控制面板项——》程序和功能&#xff0c;卸载mysql server!2.然后删除mysql文件夹下的my.ini文件及所有文件3.运行“regedit”文件&#xff0c;如图&#xff0c;打开注册表删除HKEY_LOCAL_MACHINE\SYSTEM\ControlSet001\Services\Eventlog\Application\MyS…

(转)unity web 缓存解决方案

unity web 缓存解决方案 官方发布 web版限制五十M缓存&#xff0c;根据自己的经验绕了过去&#xff0c;解决了缓存的问题。带工程&#xff0c;带源代码。由于本人的水平也有限&#xff0c;是用JS来解决的&#xff0c;如果你还是没有头绪&#xff0c;可以购买来试试。 http://it…

mysql 当前timestamp_MySQL表结构使用timestamp以自动获取当前时间

一、怀念ACCESS的时代ACCESS的表结构如果想自动插入当前时间的时候&#xff0c;可以在列定义的时候输入默认值now ()就可以实现自动插入当前时间了。而MySQL则行不通&#xff0c;MySQL这个东东默认是命令行下面的产物&#xff0c;如果不是后来有了phpMyAdmin这个图形化界面东东…

[转]如何设置win7一直以管理员身份运行

如何设置win7一直以管理员身份运行 在win7有些程序需要以管理员的身份才能运行&#xff0c;但是我们几乎天天都要运行这些程序&#xff0c;老是要手动选下很麻烦&#xff0c;怎么样设置才能这个程序以后运行都直接是以管理员的身份运行&#xff0c;不用在让我们去选择了&#x…

mysql5.7转移_mysql从5.7.17迁移到5.7.22业务报错

本帖最后由 mark_huang 于 2018-5-22 16:45 编辑mysql从5.7.17迁移到5.7.22后&#xff0c;业务半个月报错2次(报错时间毫无规律) 以下是业务报错日志&#xff0c;truncate表后重启执行程序业务恢复正常。 原库和新库的 wait_timeout都是8个小时2018-05-20 06:30:30.020 [schedu…

15. HTML 块 div span

一&#xff1a; 块元素 block level element. 内联元素 inline element. 二&#xff1a; 块元素特点&#xff1a; 块级元素在浏览器显示时&#xff0c; 通常会以新行开始和结束。 例子&#xff1a;<h1> <p> <ul> <table> 内联元素特点&…

mysql 优化 修复原理_mysql下表的修复与优化

在很多的linux生产服务器里很多时候用数据库是mysql&#xff0c;在数据库里算是轻量级的数据库&#xff0c;但是长期以来的使用往往会出现一些问题&#xff0c;一般的状况是无法读取表&#xff0c;此时我们就要来修复表、优化表&#xff0c;减小磁盘使用量&#xff0c;以方便数…

funny

var life { "work_hard","have_fun","make_history" };转载于:https://www.cnblogs.com/yqskj/p/3205328.html

MySQL substring-index_mysql函数之SUBSTRING_INDEX(str,/,-1)

SUBSTRING_INDEX的用法&#xff1a; •SUBSTRING_INDEX(str,delim,count) 在定界符 delim 以及count 出现前&#xff0c;从字符串str返回自字符串。若count为正值,则返回最终定界符(从左边开始)若为-1则是从后往前截取SELECT substring_index(Hn_P00001, P, -1) -- 结果是0000…

根据/proc/meminfo对空闲内存进行占用

#include <stdio.h>#include <sys/sysinfo.h> #include <linux/kernel.h> /* 包含sysinfo结构体信息*/ #include <unistd.h>#include <string> #include <iostream> #include <fstream> #include <map> #include <vect…

integer 负数字符串比较_JAVA源码之Integer-1

四、方法toString三个方法&#xff0c;其中两个static方法。1、public String toString()&#xff1a;该方法内部使用toString(int i)实现。2、public static String toString(int i)&#xff1a;该方法内部使用stringSize方法巧妙的获取入参的size&#xff0c;然后用getChars把…