IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 实验 线程编程-加1操作为什么会出错?【操作系统】 -> 正文阅读

[C++知识库]实验 线程编程-加1操作为什么会出错?【操作系统】

出错

在这个代码中,函数thread()有一个循环,做了简单的加1操作。 在主函数中,调用pthread_create()创建了一个线程,这个循环16次,总共创建了16个线程,运行这个程序,我们本来的期望结果是16*10000, 但是发现结果不是160000,那到底为什么?如何解决?

源代码

9-.c

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <errno.h>
#include <unistd.h>
//开后16个线程。
#define PTHREAD_NUM 16
unsigned long sum =0;
void *thread(void *arg){
	//不使用原子锁
	int i;
	for(i= 0;i <10000;i++)  
		sum+=1;

}

int main(void){
	printf("before ...sum = %lu\n",sum);
	pthread_t pthread[PTHREAD_NUM];			//被创建线程的标识
	int ret;								//接收近回值
	void *retval[PTHREAD_NUM];
	int i;
	for(i= 0; i< PTHREAD_NUM; i++){
		ret = pthread_create(&pthread[i], NULL, thread, NULL);  
		if(ret !=0){
			perror("cause:");
			printf("creat pthread %d failed.\n",i+1);
		}
	}
	for (i=0;i<PTHREAD_NUM; i++){
		pthread_join(pthread[i], &retval[i]);  
	}
	
	printf("after ... sun = %lu\n",sum);
	return	0;
}


运行结果

请添加图片描述

修改1–原子操作

我看到,加1操作实际上是由汇编指令的三条指令完成的,如果这三条指令在执行期间不被中断,一口气执行完,那就不出错了,因此,就有了原子操作的概念,所谓原子操作,就是在执行期间不可分割,要么全部执行,要么一条也不执行。 在Linux下如何进行原子操作? gcc从4.1.2开始提供了__sync_*系列的build-in函数,用于提供加减和逻辑运算的原子操作,其声明如下:

type __sync_fetch_and_add (type *ptr, type value, ...)
type __sync_fetch_and_sub (type *ptr, type value, ...)
type __sync_fetch_and_or (type *ptr, type value, ...)

...

比如,__sync_add_and_fetch()是GCC内嵌的函数,可以进行加1的原子操作,请参看__sync_fetch_and_add系列原子操作函数述

代码

9.c

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <errno.h>
#include <unistd.h>
//开后16个线程。
#define PTHREAD_NUM 16
unsigned long sum =0;
void *thread(void *arg){
	//不使用原子锁
	int i;
	for(i= 0;i <10000;i++)  
		__sync_add_and_fetch(&sum,1);

}

int main(void){
	printf("before ...sum = %lu\n",sum);
	pthread_t pthread[PTHREAD_NUM];			//被创建线程的标识
	int ret;								//接收近回值
	void *retval[PTHREAD_NUM];
	int i;
	for(i= 0; i< PTHREAD_NUM; i++){
		ret = pthread_create(&pthread[i], NULL, thread, NULL);  
		if(ret !=0){
			perror("cause:");
			printf("creat pthread %d failed.\n",i+1);
		}
	}
	for (i=0;i<PTHREAD_NUM; i++){
		pthread_join(pthread[i], &retval[i]);  
	}
	
	printf("after ... sun = %lu\n",sum);
	return	0;
}


结果

请添加图片描述

修改2–加锁

最简单的处理办法就是加锁保护,请参看
POSIX详解pthread_mutex_lock

代码

9_.c

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <errno.h>
#include <unistd.h>
//开后16个线程。
#define PTHREAD_NUM 16
unsigned long sum =0;
pthread_mutex_t mymutex=PTHREAD_MUTEX_INITIALIZER;
void *thread(void *arg){
	//不使用原子锁
	int i;
	for(i= 0;i <10000;i++){
		pthread_mutex_lock(&mymutex); 
		sum+=1;
		pthread_mutex_unlock(&mymutex);
		
	}
}

int main(void){
	printf("before ...sum = %lu\n",sum);
	pthread_t pthread[PTHREAD_NUM];			//被创建线程的标识
	int ret;								//接收近回值
	void *retval[PTHREAD_NUM];
	int i;
	for(i= 0; i< PTHREAD_NUM; i++){
		ret = pthread_create(&pthread[i], NULL, thread, NULL);  
		if(ret !=0){
			perror("cause:");
			printf("creat pthread %d failed.\n",i+1);
		}
	}
	for (i=0;i<PTHREAD_NUM; i++){
		pthread_join(pthread[i], &retval[i]);  
	}
	
	printf("after ... sun = %lu\n",sum);
	return	0;
}


结果

请添加图片描述

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-05-13 11:37:19  更:2022-05-13 11:37:24 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 19:28:47-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码