参考资料

线程的创建

线程的创建有多种方式

std::thread t1(可调用对象);

由于实现(内部的实现这里不在探讨),std::thread()创建一个新的线程可以接受任意的可调用对象类型(带参数或者不带参数),包括lambda表达式(带变量捕获或者不带),函数,函数对象,以及函数指针。

下面简单的探讨一下。

1.通过一个不带参数的函数创建线程。

#include <iostream>
#include <thread> // 多线程头文件 void Hello() {
std::cout << "Hello, World!" << std::endl;
} int main() {
// 创建一个线程对象,注意函数 Hello 将立即运行。
std::thread t(&Hello); // 等待线程结束。
// 否则线程还没执行(完),主程序就已经结束了。
t.join(); return 0;
}

cppreference 关于立即运行有提到

在启动了一个线程(创建了一个thread对象)之后,当这个线程结束的时候(std::terminate()),我们如何去回收线程所使用的资源呢?

thread库给我们两种选择:

  • 1.加入式(join())
  • 2.分离式(detach())

值得一提的是,你必须在thread对象销毁之前做出选择 这是因为线程可能在你加入或分离线程之前,就已经结束了,之后如果再去分离它,线程可能会在thread对象销毁之后继续运行下去。

2.通过一个带参数的函数创建线程。

#include <iostream>
#include <thread> void Hello(const char* what) {
// 睡眠一秒以模拟数据处理。
std::this_thread::sleep_for(std::chrono::seconds(1));
std::cout << "Hello, " << what << "!" << std::endl;
} int main() {
std::thread t(&Hello, "World"); // 等价于使用 bind:
// std::thread t(std::bind(&Hello, "World")); t.join(); return 0;
}

3.通过一个函数对象——即仿函数(functor)——创建线程。

class Hello {
public:
void operator()(const char* what) {
std::this_thread::sleep_for(std::chrono::seconds(1));
std::cout << "Hello, " << what << "!" << std::endl;
}
}; int main() {
Hello hello; // 方式一:拷贝函数对象。
std::thread t1(hello, "World");
t1.join(); // 方式二:不拷贝函数对象,通过 boost::ref 传入引用。
// 用户必须保证被线程引用的函数对象,拥有超出线程的生命期。
// 比如这里通过 join 线程保证了这一点。
std::thread t2(std::ref(hello), "World");
t2. return 0;
}

4.通过一个成员函数创建线程。

// 通过成员函数来创建线程. 注意和普通函数的区别
#include <thread>
#include <iostream> class Hello {
public:
void ThreadFuntion() {
std::this_thread::sleep_for(std::chrono::seconds(5));
std::cout << "ThreadFuntion1 " << std::this_thread::get_id() << std::endl;
}
}; int main() {
Hello hello; // 需要一个对象
std::thread t1(&Hello::ThreadFuntion, &hello);
t1.join(); return 0;
}

5.通过一个成员函数创建线程(在构造函数中操作)。

与前例不同之处在于,需要以 bind 绑定 this 指针作为第一个参数。

#include <iostream>
#include <thread> class Hello {
public:
Hello() {
std::thread t(std::bind(&Hello::Entry, this, "World"));
t.join();
} private:
// 线程函数
void Entry(const char* what) {
std::this_thread::sleep_for(std::chrono::seconds(1));
std::cout << "Hello, " << what << "!" << std::endl;
}
}; int main() {
Hello hello;
return 0;
}

6.使用lambda

#include <iostream>
#include <thread> //
int main() {
auto threadFunction = []() {
std::cout << "Lambda thread id " << std::this_thread::get_id() <<std::endl; // 打印线程id
std::cout << "使用lambda表达式作为可调用对象" << std::endl;
}; std::cout << "Main thread id " << std::this_thread::get_id() <<std::endl;
std::thread t1(threadFunction);
t1.join();
return 0;
}

join Vs detach

join

join()字面意思是连接一个线程,意味着主动地等待线程的终止。

join()是这样工作的,在调用进程中join(),当新的线程终止时,join()会清理相关的资源(any storage associated with the thread),然后返回,调用线程再继续向下执行。 正是由于join()清理了线程的相关资源,因而我们之前的thread对象与已销毁的线程就没有关系了,这意味着一个线程的对象每次你只能使用一次join(),当你调用的join()之后joinable()就将返回false了。

关于joinable()

判断是否可以成功使用join() 或者detach(), 返回值为bool .

#include <iostream>
#include <thread> void foo()
{
std::this_thread::sleep_for(std::chrono::seconds(1));
} int main()
{
std::thread t(foo);
std::cout << "before joining,joinable=" << std::boolalpha << t.joinable() << std::endl;
t.join();
std::cout << "after joining, joinable=" << std::boolalpha << t.joinable() << '\n';
} // output
// [thread]main
// before joining,joinable=true
// after joining, joinable=false

detach

分离式,对应的函数是detach()。

detach这个词的意思是分离的意思,对一个thread对象使用detach()意味着从调用线程分理出这个新的线程,我们称分离的线程叫做守护线程(daemon threads)。之后也就不能再与这个线程交互。

分离的线程会在后台运行,其所有权(ownership)和控制权将会交给c++运行库。同时,C++运行库保证,当线程退出时,其相关资源的能够正确的回收。

分离的线程,它运行结束后,不再需要通知调用它的线程

线程的标识

线程id。

类 thread::id 是轻量的可频繁复制类,它作为 std::thread 对象的唯一标识符工作。

  • std::this_thread::get_id()这个函数获取线程的标识符 (在线程的函数中)
  • 线程对象也有一个function get_id() std::thread::id get_id() const noexcept;
// source: cppreference
#include <iostream>
#include <thread>
#include <chrono> void foo()
{
// 看这里
std::cout << std::this_thread::get_id() <<std::endl;
std::this_thread::sleep_for(std::chrono::seconds(1));
} int main()
{
// 看这里
std::thread t1(foo);
std::thread::id t1_id = t1.get_id(); std::thread t2(foo);
std::thread::id t2_id = t2.get_id(); std::cout << "t1's id: " << t1_id << '\n';
std::cout << "t2's id: " << t2_id << '\n'; t1.join();
t2.join();
}

C++11多线程编程--线程创建的更多相关文章

  1. Linux多线程编程——线程的创建与退出

    POSIX线程标准:该标准定义了创建和操纵线程的一整套API.在类Unix操作系统(Unix.Linux.Mac OS X等)中,都使用Pthreads作为操作系统的线程.Windows操作系统也有其 ...

  2. C++11 多线程编程 使用lambda创建std::thread (生产/消费者模式)

    要写个tcp server / client的博客,想着先写个c++11多线程程序.方便后面写博客使用. 目前c++11中写多线程已经很方便了,不用再像之前的pthread_create,c++11中 ...

  3. java多线程编程(二创建线程)

    1.概念           因为java是完全面向对象的,所以在java中,我们说的线程,就是Thread类的一个实例对象.所以,一个线程就是一个对象,它有自己字段和方法. 2.创建线程 创建线程有 ...

  4. C++11多线程编程

    1. 多线程编程 在进行桌面应用程序开发的时候, 假设应用程序在某些情况下需要处理比较复杂的逻辑, 如果只有一个线程去处理,就会导致窗口卡顿,无法处理用户的相关操作.这种情况下就需要使用多线程,其中一 ...

  5. Java基础加强之多线程篇(线程创建与终止、互斥、通信、本地变量)

    线程创建与终止 线程创建 Thread类与Runnable接口的关系 public interface Runnable { public abstract void run(); } public ...

  6. Java基础之多线程篇(线程创建与终止、互斥、通信、本地变量)

    线程创建与终止 线程创建 Thread类与Runnable接口的关系 public interface Runnable { public abstract void run(); } public ...

  7. Python多线程之线程创建和终止

    python主要是通过thread和threading这两个模块来实现多线程支持. python的thread模块是比較底层的模块,python的threading模块是对thread做了一些封装,能 ...

  8. Linux C多线程编程-线程互斥

    Linux下的多线程编程需要注意的是程序需要包含头文件pthread.h,在生成可执行文件的时候需要链接库libpthread.a或者libpthread.so. 线程创建函数: pthread_cr ...

  9. win32多线程: 线程创建与结束等待

    #include<Windows.h> #include<iostream> using namespace std; /*1.在启动一个线程之前,必须为线程编写一个全局的线程 ...

随机推荐

  1. java小项目——抽奖系统

    来了来了!这不又到考试周了吗!愁人,又得复习,复习,复习!这段时间每天都在复习线代和高数!(说是复习,说实话其实是在预习,啊哈哈哈哈哈),得有一段时间都没有学到新的知识了,代码感觉都生疏了,惆怅.博客 ...

  2. Redis - (Linux)安装与配置

    Redis 是完全开源免费的,遵守BSD协议,是一个高性能的key-value数据库. Redis 与其他 key - value 缓存产品有以下三个特点: 1:Redis支持数据的持久化,可以将内存 ...

  3. Windows下6款实用软件(强烈推荐!)

    Windows下6款实用软件 1.notepads Notepads作为一款编辑器,美观.轻量,功能强大,支持多标签页.Markdown.日常文本编辑.查看,Notepads轻松胜任,如果厌烦了Win ...

  4. CODING DevOps 系列第五课:微服务测试——微服务下展开体系化的微服务测试

    微服务测试的痛点与挑战 这张图可以形象地展示单体服务和微服务的对比,单体应用就像左边巨大的集装箱,软件模块和应用都包括其中:而微服务就像是由一个小集装箱组成,微小的服务组成一个庞大.完整的系统.单体服 ...

  5. new jup在新一代中存在

    1.灰度发布服务动态路由 动态配置路由规则,实现对调用流量的精确控制.可配置基于版本.IP.自定义标签等复杂的规则.2.服务鉴权示例2需求:服务 provider-demo 只允许来自 consume ...

  6. IntelliJ IDEA安装配置、搭建Spring MVC

    安装前必备软件: 1.jdk1.8.0_144安装包 2.IntelliJ IDEA 2016.1.1(64) 3.Tomcat安装包 4.Mysql.MySQL-JDBC驱动安装包 5.Jetbra ...

  7. php抽奖功能

    在项目开发中经常会遇到花钱抽奖类型的需求.但是老板总是担心用户用小钱抽到大奖.这样会导致项目亏损.下边这段代码可以有效制止抽奖项目亏钱. 个人奖池: 语言:thinkphp redis mysql 表 ...

  8. C#状态机Stateless

    最近在折腾一些控制相关的软件设计,想起来状态机这个东西,对解决一些控制系统状态切换还是挺有用的. 状态机(有限状态自动机)网上有很多介绍.简单理解就是定义一系列状态,通过一系列的事件,可以使得状态可以 ...

  9. 关于数据文件的文件头1-P2

    文章目录 1 疑问点 2 问题模拟 2.1 dump 0,1块 2.2 查看trc文件 2.3 如何查看 1 疑问点 这里引用p2处的一段话: 事实上,每个文件的前128个块,都是文件头,被Oracl ...

  10. 查看windows操作系统的默认编码

    转自:https://blog.csdn.net/zp357252539/article/details/79084480/ 在Windows平台下,进入DOS窗口,输入:chcp 可以得到操作系统的 ...