当前课程知识点:基于Linux的C++ >  第十四讲 线程编程 >  14.5 线程管理(三) >  LinuxCPP1405

返回《基于Linux的C++》慕课在线视频课程列表

LinuxCPP1405在线视频

LinuxCPP1405

下一节:LinuxCPP1406

返回《基于Linux的C++》慕课在线视频列表

LinuxCPP1405课程教案、知识点、字幕

关于线程的ID有两个函数你可以用

第一个是pthread_equal()

判定两个线程是不是相等

第二个是pthread_self()

返回当前线程的ID

有的时候我们对当前线程的ID

会特别的关注

因为我们要区分我一个程序中

启动的很多个线程

它到底哪一个是哪一个 对吧

所以这两个函数你都可以用

在创建线程的时候

可以通过设置线程的属性

来精确地调整线程的行为

设置线程的属性过程

有一个固定的流程

和以前我们讲进程间通讯的时候

它的很多属性的设置都非常得类似

首先你要创建一个pthread_attr_t

这种型的一个对象

然后你调用pthread_attr_init()这样的函数

来初始化线程属性对象的一个缺省设置

做完了这个之后

你就可以精细地

去调整它的属性中的每一个选项了

设定完线程的属性

你再调用pthread_create()

创建这个线程的时候

就可以传递

这个线程属性对象的地址进去

它就会使用你设定好的

这个线程属性去创建线程

如果在线程创建完了之后

这个线程属性对象你不要了

那你就可以调用pthread_attr_destroy()

销毁它

销毁这个动作仅仅是销毁

线程属性对象里边的属性设置

它并没有清除这个对象本身

所以你可以重新地调用

pthread_attr_init()去初始化它

再次初始化 再次使用它

这个是没有关系的

有一点是需要说明的

单一的一个线程属性对象

可以用来创建很多个线程

这个没关系

对它来说 它只是一个线程属性对象

它只需要从中获取线程创建时候的

一些特定属性值

能够得到这个值就行

所以从某种程度来讲一个

单一的一个线程属性对象

可以构造很多很多个线程 这是一个

第二个 线程创建以后

继续保留这个线程属性对象本身

是没有意义的

所以大部分时候

我们都会destroy它 销毁它

对于大多数Linux程序来说

线程属性对象中最重要的那个属性

就是它的分离状态

其它我们很少会设定

一个Linux线程

从分类的角度来讲 是分成两大类

一类是可联线程

一类我们称之为分离线程

什么是可联线程呢

就是这样的一个线程在中止的时候

并不去做自动地清除工作

这个时候就会导致这样的线程

就像僵尸进程一样结束了

但是它的状态并没有被取出来

主线程必须调用join那个函数

来等待它结束

才能去完成它最后的清除工作

这个就叫joinable

所以叫可联线程 你要联接它

然后才能做最后的清除工作

第二种就是你不需要去做清除工作的

那么这就叫分离线程

你把这个线程设置成一个分离的

那么这个线程做完了

它自动地就被清除掉了

就什么你不用管了

主线程就不用管它了

就这个意思

缺省设置是可联线程

主线程必须调用pthread_join()

去等待它结束

这是很典型的线程的分类

就这么两种

一个可联线程可以通过pthread_detach()

这样的一个函数调用把它分离

但是你分离过了以后

这个线程就不可以再次联接了

你再想把它联接进来那就不行了

detach这线程一旦被分离了

像这样的线程一旦调用了pthread_detach()

这样的函数分离了它

那么你就再也不能调用pthread_join()

去等待它了

这一点是需要特别注意的

所有的线程缺省的设置都是可联线程

你都必须调用pthread_join()

去等待它结束了

有两个函数可以用来设置或获取

线程的分离状态

一个是pthread_attr_setdetachstate()

这是设置线程的分离属性

还有一个函数叫pthread_attr_getdetachstate()

这个就是获取线程的分离属性

我们最关心的是那个set函数

你传一个线程属性对象的一个指针

然后你传一个

它的分离状态参数进去 就OK了

缺省情况下它不是一个可联线程吗

所以如果你在创建线程的最初

就把它变成一个分离线程

那么你的第二个参数你就传

PTHREAD_CREATE_DETACHED

就是创建的时候

就让这个线程处于分离状态

以后你就再不用等待结束了

它自己结束 它就over

就调用这个函数就OK了

我们看这样一个例子

在主函数里

我们会定义一个线程属性对象attr

然后呢调用pthread_attr_init()初始化它

完成初始化以后我们调用

pthread_attr_setdetachstate()

传分离线程标志

把这个属性给它传进去 设定好它

然后你就可以调用pthread_create()

去创建这个线程

然后这个线程就会运行

完事以后

这个属性对象 我们就调用

pthread_attr_destroy()销毁

就完了 很简单吧

就是一个基本属性的设置

你完成它的设置 就OK了

只不过这里边有一系列的属性

每个可能需要做的设置

你都需要查阅帮助手册

然后才能知道我为什么要做这个设置

如果我需要某种特定的设置

我应该设置什么样的值

我们的课堂上

不可能事无巨细 面面俱到 对吧

所以这些地方你知道怎么用就行了

具体的细节在需要的时候查阅帮助手册

有一个函数叫pthread_cancel()

用来负责撤销一个线程

这个函数特别简单

就传一个线程ID进去就完了

已撤销的线程可以联接

实际上 我们也必须联接这个线程

然后释放它的资源

你不联接的话

它的好多最后的清除工作就没法做

这个线程被撤销了

那么你就联接它 释放它的资源

除非 它是一个分离线程

那你就不能管了

这是线程的撤销

线程的撤销 它有自己的类型或状态

就三种 一个是异步可撤销

就在它线程执行的任何时候

你都可以把它撤销掉

这叫异步可撤销

第二种就是同步可撤销

就是线程你可以撤销

但是这个撤销呢

必须要等到一些特定的撤销点

你才能去做

所有的这些撤销操作形成一个队列之后

你要入队 去排队的

到了一个特定的撤销点

然后你才去做撤销的动作

所以它是一种延迟的撤销

还有一种当然就是不可撤销

这个线程本身是不能够被撤销的

所有对这个线程发出的撤销请求

它都会被系统自动地忽略

线程是不管的

可以通过这个函数

pthread_setcanceltype() 来设定

最重要的两个参数

就是PTHREAD_CANCEL_ASYNCHRONOUS

这是一个异步可撤销

还有PTHREAD_CANCEL_DEFERRED

defer叫延迟

延迟可撤销

其实就是同步可撤销

要到固定的撤销点它才会撤销

这就叫同步可撤销

用pthread_setcancelstate()

来设置线程的撤销状态

刚才做的是设置线程的撤销类型

现在我们使用pthread_setcancelstate()

来设置线程的撤销状态

这是决定这个线程能不能够被撤销

刚才是以什么样的方式撤销

这是能不能够撤销

所以PTHREAD_CANCEL_ENABLE

说明这个线程是可撤销的

就注意这一点

这个函数pthread_testcancel()

用于设置对应的撤销点

它如果设置成功了这个撤销点

到了这个撤销点

那么当发现这个线程是属于

延迟可撤销的 也就是同步可撤销的

那么它就会撤销那个线程

所以在编程的时候

要特别注意这一点 就是什么呢

就是当你有一些线程

需要以一个同步可撤的模式

进行撤销的时候

那么你就要周期性地

在你的函数内部调用这个函数

设定一些撤销点

这样的话能保证这个线程在完结了之后

尽快地会被撤销

做最后的清除工作

在编程的时候有一个技术细节

就是我们可以使用这个线程的撤销状态

来构造一个临界区

什么叫临界区呢

临界区就是一个关键性的一个区域

在这个关键性区域里边的代码

要么就全部执行 要么就一条都不做

那么你就可以设置

这个线程的撤销状态

这个线程一旦进入这个临界区

你就把它设置成一个不可撤销的这个模式

那么只能等它离开临界区

你恢复它一个可撤销模式之后

线程才能够被撤销

这样的话

这段代码就保证它一定会被执行

或者一定不被执行

这就完成了临界区的设计

我们看它的具体的示例

实际内容本身很简单

我们这里边想做一个transfer

想做一个转账

一个from这个账户转到to的这个账户

把一部分钱从一个from账户

转到to这个账户

当然这是一个账户的数组

from和to都是这个账户的

数组里边的下标

就从一个账户转到另外一个账户

我想转一笔钱过去

这笔钱转账嘛 因为一边账户要减

另外一边账户要加

所以这个动作应该要么都做

要么都不做

不能一边减了 那边没加

那不就完蛋了嘛 数据就不一致了

所以这个地方就需要特别注意

所以这个操作应该做成一个临界区

那么当我们构造这个线程的时候

就可以在进入这个临界区之前

调用pthread_setcancelstate()

说明我不允许你撤销我这个线程

接下来执行的代码

然后调用pthread_setcancelstate()

重新恢复它原来的线程状态

这样的话就保证我这段代码

要么就没做 要么就一定被做完了

这就是线程的撤销

基于Linux的C++课程列表:

第一讲 C/C++基本语法元素

-1.1 提纲

--LinuxCPP0101

-1.2 程序设计的基本概念

--LinuxCPP0102

-1.3 简单C/C++程序介绍

--LinuxCPP0103

-1.4 程序设计的基本流程

--LinuxCPP0104

-1.5 基本语法元素

--LinuxCPP0105

-1.6 程序设计风格

--LinuxCPP0106

-1.7 编程实践

--LinuxCPP0107

-第一讲 C/C++基本语法元素--编程实践提交入口

第二讲 程序控制结构

-2.1 提纲

--LinuxCPP0201

-2.2 结构化程序设计基础

--LinuxCPP0202

-2.3 布尔数据

--LinuxCPP0203

-2.4 分支结构

--LinuxCPP0204

-2.5 break语句

--LinuxCPP0205

-2.6 循环结构

--LinuxCPP0206

-2.7 编程实践

--LinuxCPP0207

-第二讲 程序控制结构--编程实践提交入口

第三讲 函数

-3.1 提纲

--LinuxCPP0301

-3.2 函数声明、调用与定义

--LinuxCPP0302

-3.3 函数调用栈框架

--LinuxCPP0303

-3.4 编程实践

--LinuxCPP0304

-第三讲 函数--编程实践提交入口

第四讲 算法

-4.1 提纲

--LinuxCPP0401

-4.2 算法概念与特征

--LinuxCPP0402

-4.3 算法描述

--LinuxCPP0403

-4.4 算法设计与实现

--LinuxCPP0404

-4.5 递归算法(一)

--LinuxCPP0405

-4.6 递归算法(二)

--LinuxCPP0406

-4.7 容错与计算复杂度

--LinuxCPP0407

-4.8 编程实践

--LinuxCPP0408

-第四讲 算法--编程实践提交入口

第五讲 程序组织与开发方法

-5.1 提纲

--LinuxCPP0501

-5.2 库与接口

--LinuxCPP0502

-5.3 随机数库(一)

--LinuxCPP0503

-5.4 随机数库(二)

--LinuxCPP0504

-5.5 作用域与生存期

--LinuxCPP0505

-5.6 典型软件开发流程(一)

--LinuxCPP0506

-5.7 典型软件开发流程(二)

--LinuxCPP0507

-5.8 编程实践

--LinuxCPP0508

-第五讲 程序组织与开发方法--编程实践提交入口

第六讲 复合数据类型

-6.1 提纲

--LinuxCPP0601

-6.2 字符

--LinuxCPP0602

-6.3 数组(一)

--LinuxCPP0603

-6.4 数组(二)

--LinuxCPP0604

-6.5 结构体

--LinuxCPP0605

-6.6 编程实践

--LinuxCPP0606

-第六讲 复合数据类型--编程实践提交入口

第七讲 指针与引用

-7.1 提纲

--LinuxCPP0701

-7.2 指针基本概念

--LinuxCPP0702

-7.3 指针与函数

--LinuxCPP0703

-7.4 指针与复合数据类型(一)

--LinuxCPP0704

-7.5 指针与复合数据类型(二)

--LinuxCPP0705

-7.6 字符串

--LinuxCPP0706

-7.7 动态存储管理(一)

--LinuxCPP0707

-7.8 动态存储管理(二)

--LinuxCPP0708

-7.9 引用

--LinuxCPP0709

-7.10 编程实践

--LinuxCPP0710

-第七讲 指针与引用--编程实践提交入口

第八讲 链表与程序抽象

-8.1 提纲

--LinuxCPP0801

-8.2 数据抽象(一)

--LinuxCPP0802

-8.3 数据抽象(二)

--LinuxCPP0803

-8.4 链表(一)

--LinuxCPP0804

-8.5 链表(二)

--LinuxCPP0805

-8.6 链表(三)

--LinuxCPP0806

-8.7 链表(四)

--LinuxCPP0807

-8.8 函数指针(一)

--LinuxCPP0808

-8.9 函数指针(二)

--LinuxCPP0809

-8.10 抽象链表(一)

--LinuxCPP0810

-8.11 抽象链表(二)

--LinuxCPP0811

-8.12 编程实践

--LinuxCPP0812

-第八讲 链表与程序抽象--编程实践提交入口

第九讲 类与对象

-9.1 提纲

--LinuxCPP0901

-9.2 程序抽象与面向对象

--LinuxCPP0902

-9.3 类类型

--LinuxCPP0903

-9.4 对象(一)

--LinuxCPP0904

-9.5 对象(二)

--LinuxCPP0905

-9.6 类与对象的成员(一)

--LinuxCPP0906

-9.7 类与对象的成员(二)

--LinuxCPP0907

-9.8 类与对象的成员(三)

--LinuxCPP0908

-9.9 继承(一)

--LinuxCPP0909

-9.10 继承(二)

--LinuxCPP0910

-9.11 继承(三)

--LinuxCPP0911

-9.12 多态(一)

--LinuxCPP0912

-9.13 多态(二)

--LinuxCPP0913

-9.14 编程实践

--LinuxCPP0914

-第九讲 类与对象--编程实践提交入口

第十讲 操作符重载

-10.1 提纲

--LinuxCPP1001

-10.2 四则运算符重载(一)

--LinuxCPP1002

-10.3 四则运算符重载(二)

--LinuxCPP1003

-10.4 关系与下标操作符重载

--LinuxCPP1004

-10.5 赋值操作符重载(一)

--LinuxCPP1005

-10.6 赋值操作符重载(二)

--LinuxCPP1006

-10.7 赋值操作符重载(三)

--LinuxCPP1007

-10.8 赋值操作符重载(四)

--LinuxCPP1008

-10.9 赋值操作符重载(五)

--LinuxCPP1009

-10.10 流操作符重载(一)

--LinuxCPP1010

-10.11 流操作符重载(二)

--LinuxCPP1011

-10.12 流操作符重载(三)

--LinuxCPP1012

-10.13 操作符重载总结

--LinuxCPP1013

-10.14 编程实践

--LinuxCPP1014

-第十讲 操作符重载--编程实践提交入口

第十一讲 泛型编程

-11.1 提纲

--LinuxCPP1101

-11.2 泛型编程概览

--LinuxCPP1102

-11.3 异常处理机制(一)

--LinuxCPP1103

-11.4 异常处理机制(二)

--LinuxCPP1104

-11.5 运行期型式信息(一)

--LinuxCPP1105

-11.6 运行期型式信息(二)

--LinuxCPP1106

-11.7 模板与型式参数化

--LinuxCPP1107

-11.8 题外话:术语翻译

--LinuxCPP1108

-11.9 泛型编程实践(一)

--LinuxCPP1109

-11.10 泛型编程实践(二)

--LinuxCPP1110

-11.11 泛型编程实践(三)

--LinuxCPP1111

-11.12 泛型编程实践(四)

--LinuxCPP1112

-11.13 泛型编程实践(五)

--LinuxCPP1113

-11.14 泛型编程实践(六)

--LinuxCPP1114

-11.15 泛型编程实践(七)

--LinuxCPP1115

-11.16 泛型编程实践(八)

--LinuxCPP1116

-11.17 泛型编程实践(九)

--LinuxCPP1117

-11.18 泛型编程实践(十)

--LinuxCPP1118

-11.19 编程实践

--LinuxCPP1119

-第十一讲 泛型编程--编程实践提交入口

第十二讲 Linux系统编程基础

-12.1 提纲

--LinuxCPP1201

-12.2 程序执行环境(一)

--LinuxCPP1202

-12.3 程序执行环境(二)

--LinuxCPP1203

-12.4 程序执行环境(三)

--LinuxCPP1204

-12.5 程序执行环境(四)

--LinuxCPP1205

-12.6 输入输出(一)

--LinuxCPP1206

-12.7 输入输出(二)

--LinuxCPP1207

-12.8 文件系统

--LinuxCPP1208

-12.9 设备

--LinuxCPP1209

-12.10 库(一)

--LinuxCPP1210

-12.11 库(二)

--LinuxCPP1211

-12.12 makefile文件(一)

--LinuxCPP1212

-12.13 makefile文件(二)

--LinuxCPP1213

-12.14 makefile文件(三)

--LinuxCPP1214

-12.15 编程实践

--LinuxCPP1215

-第十二讲 Linux系统编程基础--编程实践提交入口

第十三讲 进程编程

-13.01 提纲

--LinuxCPP1301

-13.02 进程基本概念

--LinuxCPP1302

-13.03 信号

--LinuxCPP1303

-13.04 进程管理(一)

--LinuxCPP1304

-13.05 进程管理(二)

--LinuxCPP1305

-13.06 进程管理(三)

--LinuxCPP1306

-13.07 进程间通信(一)

--LinuxCPP1307

-13.08 进程间通信(二)

--LinuxCPP1308

-13.09 进程间通信(三)

--LinuxCPP1309

-13.10 进程间通信(四)

--LinuxCPP1310

-13.11 进程池

--LinuxCPP1311

-13.12 编程实践

--LinuxCPP1312

-第十三讲 进程编程--编程实践提交入口

第十四讲 线程编程

-14.1 提纲

--LinuxCPP1401

-14.2 线程基本概念

--LinuxCPP1402

-14.3 线程管理(一)

--LinuxCPP1403

-14.4 线程管理(二)

--LinuxCPP1404

-14.5 线程管理(三)

--LinuxCPP1405

-14.6 线程管理(四)

--LinuxCPP1406

-14.7 线程同步机制(一)

--LinuxCPP1407

-14.8 线程同步机制(二)

--LinuxCPP1408

-14.9 C++11线程库(一)

--LinuxCPP1409

-14.10 C++11线程库(二)

--LinuxCPP1410

-14.11 C++11线程库(三)

--LinuxCPP1411

-14.12 C++11线程库(四)

--LinuxCPP1412

-14.13 C++11线程库(五)

--LinuxCPP1413

-14.14 编程实践

--LinuxCPP1414

-第十四讲 线程编程--编程实践提交入口

第十五讲 网络编程

-15.1 提纲

--LinuxCPP1501

-15.2 Internet网络协议

--LinuxCPP1502

-15.3 套接字(一)

--LinuxCPP1503

-15.4 套接字(二)

--LinuxCPP1504

-15.5 编程实践

--LinuxCPP1505

-第十五讲 网络编程--编程实践提交入口

课程文档

-课程PDF文件

LinuxCPP1405笔记与讨论

也许你还感兴趣的课程:

© 柠檬大学-慕课导航 课程版权归原始院校所有,
本网站仅通过互联网进行慕课课程索引,不提供在线课程学习和视频,请同学们点击报名到课程提供网站进行学习。