一、C/C++多线程操作说明

C/C++多线程基本操作如下: 
1. 线程的建立结束 
2. 线程的互斥和同步 
3. 使用信号量控制线程 
4. 线程的基本属性配置

在C/C++代码编写时,使用多线程机制,首先需要做的事情就是声明引用,具体如下:

#include "pthread.h"

二、线程基本操作方法

基本线程操作:

1. pthread_create():创建线程开始运行相关线程函数,运行结束则线程退出

2. pthread_eixt():因为exit()是用来结束进程的,所以则需要使用特定结束线程的函数

3. pthread_join():挂起当前线程,用于阻塞式地等待线程结束,如果线程已结束则立即返回,0=成功

4. pthread_cancel():发送终止信号给thread线程,成功返回0,但是成功并不意味着thread会终止

5. pthread_testcancel():在不包含取消点,但是又需要取消点的地方创建一个取消点,以便在一个没有包含取消点的执行代码线程中响应取消请求.

6. pthread_setcancelstate():设置本线程对cancle线程的反应

7. pthread_setcanceltype():设置取消状态 继续运行至下一个取消点再退出或者是立即执行取消动作

8. pthread_setcancel():设置取消状态

三、线程互斥与同步机制

基本的互斥与同步的操作方法:

1. pthread_mutex_init():互斥锁的初始化

2. pthread_mutex_lock():锁定互斥锁,如果尝试锁定已经被上锁的互斥锁则阻塞至可用为止

3. pthread_mutex_trylock():非阻塞的锁定互斥锁

4. pthread_mutex_unlock():释放互斥锁

5. pthread_mutex_destory():互斥锁销毁函数

四、多线程实践

1. 基本的线程及建立运行

下面的代码是C/C++开发的基本的线程的运行,使用的就是最基本的pthread.h:

/* thread.c */
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h> #define THREAD_NUMBER 3 /*线程数*/
#define REPEAT_NUMBER 5 /*每个线程中的小任务数*/
#define DELAY_TIME_LEVELS 10.0 /*小任务之间的最大时间间隔*/
//
void *thrd_func(void *arg) {
/* 线程函数例程 */
int thrd_num = (int)arg;
int delay_time = ;
int count = ;
printf("Thread %d is starting\n", thrd_num);
for (count = ; count < REPEAT_NUMBER; count++) {
delay_time = (int)(rand() * DELAY_TIME_LEVELS/(RAND_MAX)) + ;
sleep(delay_time);
printf("\tThread %d: job %d delay = %d\n", thrd_num, count, delay_time);
} printf("Thread %d finished\n", thrd_num);
pthread_exit(NULL);
} int main(void) {
pthread_t thread[THREAD_NUMBER];
int no = , res;
void * thrd_ret;
srand(time(NULL));
for (no = ; no < THREAD_NUMBER; no++) {
/* 创建多线程 */
res = pthread_create(&thread[no], NULL, thrd_func, (void*)no);
if (res != ) {
printf("Create thread %d failed\n", no);
exit(res);
}
} printf("Create treads success\n Waiting for threads to finish...\n");
for (no = ; no < THREAD_NUMBER; no++) {
/* 等待线程结束 */
res = pthread_join(thread[no], &thrd_ret);
if (!res) {
printf("Thread %d joined\n", no);
} else {
printf("Thread %d join failed\n", no);
}
}
return ;
}

例程中循环3次建立3条线程,并且使用pthread_join函数依次等待线程结束; 
线程中使用rand()获取随机值随机休眠5次,随意会出现后执行的线程先执行完成; 
运行结果:

$ gcc thread.c -lpthread
$ ./a.out
Create treads success
Waiting for threads to finish...
Thread is starting
Thread is starting
Thread is starting
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread finished
Thread : job delay =
Thread : job delay =
Thread finished
Thread joined
Thread joined
Thread : job delay =
Thread finished
Thread joined

可以看到,线程1先于线程0执行,但是pthread_join的调用时间顺序,先等待线程0执行; 
由于线程1已经早结束,所以线程0被pthread_join等到的时候,线程1已结束,就在等待到线程1时,直接返回;

2. 线程执行的互斥和同步pthread_mutex_lock

下面我们在上面的程序中增加互斥锁:

/*thread_mutex.c*/
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <unistd.h> #define THREAD_NUMBER 3 /* 线程数 */
#define REPEAT_NUMBER 3 /* 每个线程的小任务数 */
#define DELAY_TIME_LEVELS 10.0 /*小任务之间的最大时间间隔*/
pthread_mutex_t mutex; void *thrd_func(void *arg) {
int thrd_num = (int)arg;
int delay_time = , count = ;
int res;
/* 互斥锁上锁 */
res = pthread_mutex_lock(&mutex);
if (res) {
printf("Thread %d lock failed\n", thrd_num);
pthread_exit(NULL);
}
printf("Thread %d is starting\n", thrd_num);
for (count = ; count < REPEAT_NUMBER; count++) {
delay_time = (int)(rand() * DELAY_TIME_LEVELS/(RAND_MAX)) + ;
sleep(delay_time);
printf("\tThread %d: job %d delay = %d\n",
thrd_num, count, delay_time);
}
printf("Thread %d finished\n", thrd_num);
/****互斥锁解锁***/
pthread_mutex_unlock(&mutex);
pthread_exit(NULL);
} int main(void) {
pthread_t thread[THREAD_NUMBER];
int no = , res;
void * thrd_ret; srand(time(NULL));
/* 互斥锁初始化 */
pthread_mutex_init(&mutex, NULL);
for (no = ; no < THREAD_NUMBER; no++) {
res = pthread_create(&thread[no], NULL, thrd_func, (void*)no);
if (res != ) {
printf("Create thread %d failed\n", no);
exit(res);
}
}
printf("Create treads success\n Waiting for threads to finish...\n");
for (no = ; no < THREAD_NUMBER; no++) {
res = pthread_join(thread[no], &thrd_ret);
if (!res) {
printf("Thread %d joined\n", no);
} else {
printf("Thread %d join failed\n", no);
}
}
pthread_mutex_destroy(&mutex);
return ;
}

在上面的例程中直接添加同步锁pthread_mutex_t; 
在线程中加入,程序在执行线程程序时,调用pthread_mutex_lock上锁,发现上锁时候后进入等待,等待锁再次释放后重新上锁; 
所以线程程序加载到队列中等待,等待成功上锁后继续执行程序代码; 
运行结果如下:

Create treads success
Waiting for threads to finish...
Thread is starting
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread finished
Thread is starting
Thread joined
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread finished
Thread is starting
Thread : job delay =
Thread : job delay =
Thread : job delay =
Thread finished
Thread joined
Thread joined 

C/C++ 多线程机制的更多相关文章

  1. Java的多线程机制系列:不得不提的volatile及指令重排序(happen-before)

    一.不得不提的volatile volatile是个很老的关键字,几乎伴随着JDK的诞生而诞生,我们都知道这个关键字,但又不太清楚什么时候会使用它:我们在JDK及开源框架中随处可见这个关键字,但并发专 ...

  2. Java的多线程机制系列:(一)总述及基础概念

    前言 这一系列多线程的文章,一方面是个人对Java现有的多线程机制的学习和记录,另一方面是希望能给不熟悉Java多线程机制.或有一定基础但理解还不够深的读者一个比较全面的介绍,旨在使读者对Java的多 ...

  3. Python GIL 多线程机制 (C source code)

    最近阅读<Python源码剖析>对进程线程的封装解释: GIL,Global Interpreter Lock,对于python的多线程机制非常重要,其如何实现的?代码中实现如下: 指向一 ...

  4. Java的多线程机制系列:(四)不得不提的volatile及指令重排序(happen-before)

    一.不得不提的volatile volatile是个很老的关键字,几乎伴随着JDK的诞生而诞生,我们都知道这个关键字,但又不太清楚什么时候会使用它:我们在JDK及开源框架中随处可见这个关键字,但并发专 ...

  5. Java的多线程机制系列:(三)synchronized的同步原理

    synchronized关键字是JDK5之实现锁(包括互斥性和可见性)的唯一途径(volatile关键字能保证可见性,但不能保证互斥性,详细参见后文关于vloatile的详述章节),其在字节码上编译为 ...

  6. python多线程机制

    Python中的线程从一开始就是操作系统的原生线程.而Python虚拟机也同样使用一个全局解释器锁(Global Interpreter Lock,GIL)来互斥线程多Python虚拟机的使用. GI ...

  7. 《Exploring in UE4》多线程机制详解[原理分析]

    转自:https://zhuanlan.zhihu.com/c_164452593 目录一.概述二."标准"多线程三.AsyncTask系统3.1 FQueuedThreadPoo ...

  8. 沉淀再出发:再谈java的多线程机制

    沉淀再出发:再谈java的多线程机制 一.前言 自从我们学习了操作系统之后,对于其中的线程和进程就有了非常深刻的理解,但是,我们可能在C,C++语言之中尝试过这些机制,并且做过相应的实验,但是对于ja ...

  9. Java的多线程机制系列:(二)缓存一致性和CAS

    一.总线锁定和缓存一致性 这是两个操作系统层面的概念.随着多核时代的到来,并发操作已经成了很正常的现象,操作系统必须要有一些机制和原语,以保证某些基本操作的原子性.首先处理器需要保证读一个字节或写一个 ...

随机推荐

  1. Zookeeper到底是干嘛的

    在Zookeeper的官网上有这么一句话:ZooKeeper is a centralized service for maintaining configuration information, n ...

  2. 在MyEclipse中使用spring-boot+mybatis+freemarker实现基本的增删改查

    一.基本环境 二.创建实体类 1.User.java package bjredcross.rainbowplans.model; import bjredcross.rainbowplans.com ...

  3. vue间通信

    1,父子组件通信 props 传递 父组件: 子组件: 2,子父组件通信 父组件: 子组件: 3,子组件与子组件传递 使用bus.js  如下 传递子组件:  接收子组件

  4. 【aardio】]SQL创建、读写 excel

    import access; var db,err = access( "/test.xls" ) //文件不存在可自动创建 //创建表 if( ! db.existsTable( ...

  5. 一次HTTP请求响应涉及了哪些?

    HTTP请求和响应步骤 TCP/IP协议 TCP三次握手 HTTP协议 HTTP请求报文 HTTP响应报文 TCP四次挥手 HTTP请求和响应步骤 以上完整表示了HTTP请求和响应的7个步骤,下面从T ...

  6. Win10下 usart驱动PL2303无法安装的问题

    随着系统的 普及,很多小伙伴也放弃了原有的win7系统,加入了win10的行列.但是相对win7的稳定来说,win10还存在很多的不足 . 新买了一个usart的模块,但是在自家的电脑上使用的时候 一 ...

  7. Appium+Python自动化 2 定位元素方式

    1.找到 Android SDK安装路径tools 下面的 uiautomatorviewer.bat,如下截图 2.点击uiautomatorviewer.bat进行启动,左上角一共四个按钮,作用分 ...

  8. docker私库Harbor部署(转载)

    系统环境 centos7.3docker-ce docker version: 18.03.0docker-compose version: 1.21.0 Install Docker CE 安装依赖 ...

  9. 死锁问题------------------------INSERT ... ON DUPLICATE KEY UPDATE*(转)

    前言    我们在实际业务场景中,经常会有一个这样的需求,插入某条记录,如果已经存在了则更新它如果更新日期或者某些列上的累加操作等,我们肯定会想到使用INSERT ... ON DUPLICATE K ...

  10. java+phantomjs实现动态网页抓取

    1.下载地址:http://phantomjs.org/download.html 2.java代码 public void getHtml(String url) { HTML="&quo ...