create_thread Ex

  • 1.导入#include—用于创建线程

  • 2.导入#include–用于时间延时 获取时间之类的

定义一个线程对象t1,这就自动创建了一个线程,参数就是你要线程去执行的函数,t1是变量名字 随便取

std::thread t1(func);

下面这里返回一个毫秒级别的时间间隔参数值,间隔10毫秒

std::chrono::milliseconds(10)
this_thread::sleep_for()就是让此线程休眠,可以传入休眠的时间
this_thread::sleep_for(std::chrono::milliseconds(10));让本线程休眠10毫秒

thread4

  • join()就是阻塞线程 阻塞的目的就是让Main主线程等待一下创建的线程,免得我函数还在跑,程序就直接结束了。
  • detach()将线程与线程对象分离

thread5

c++11还提供了一个lock_guard类,它利用了RAII机制可以保证安全释放mutex。

在std::lock_guard对象构造时,传入的mutex对象(即它所管理的mutex对象)会被当前线程锁住。

在lock_guard对象被析构时,它所管理的mutex对象会自动解锁,不需要程序员手动调用lock和unlock对mutex进行上锁和解锁操作。

lock_guard对象并不负责管理mutex对象的生命周期,lock_guard对象只是简化了mutex对象的上锁和解锁操作,方便线程对互斥量上锁,即在某个lock_guard对象的生命周期内,它所管理的锁对象会一直保持上锁状态;

而lock_guard的生命周期结束之后,它所管理的锁对象会被解锁。程序员可以非常方便地使用lock_guard,而不用担心异常安全问题。


thread6

c++11还提供了std::unique_lock

类 unique_lock 是通用互斥包装器,允许延迟锁定、锁定的有时限尝试、递归锁定、所有权转移和与条件变量一同使用。
unique_lock比lock_guard使用更加灵活,功能更加强大。
使用unique_lock需要付出更多的时间、性能成本。

thread7

互斥量很像的条件变量的知识

条件变量std::condition_variable的使用 std::condition_variable 是为了解决死锁而生的。 当互斥操作不够用而引入的。

比如,线程可能需要等待某个条件为真才能继续执行,而一个忙等待循环中可能会导致所有其他线程都无法进入临界区使得条件为真时,就会发生死锁。所以,condition_variable实例被创建出现主要就是用于唤醒等待线程从而避免死锁。

std::condition_variable:

  • notify_one()用于唤醒一个线程;
  • notify_all() 则是通知所有线程。

C++11中的std::condition_variable就像Linux下使用pthread_cond_wait和pthread_cond_signal一样,可以让线程休眠,直到被唤醒,现在在从新执行。线程等待在多线程编程中使用非常频繁,经常需要等待一些异步执行的条件的返回结果。


thread8

原子变量的使用

在新标准C++11,引入了原子操作的概念,原子操作更接近内核,并通过这个新的头文件提供了多种原子操作数据类型.

例如,atomic_bool,atomic_int等等,如果我们在多个线程中对这些类型的共享资源进行操作,编译器将保证这些操作都是原子性的,也就是说,确保任意时刻只有一个线程对这个资源进行访问,编译器将保证,多个线程访问这个共享资源的正确性。从而避免了锁的使用,提高了效率。


thread9

future与promise的使用

在c++11中增加的线程库很方便的让我们去使用线程,但是因为做出了一些改变,我们并不能像往常一样直接使用thread.join()获取线程函数的返回值了,而我们有时候又确实要利用线程函数的返回值。

thread库提供了future用来访问异步操作的结果,因为一个异步操作的结果往往不能立即获取,只能在未来的某个时候从某个地方获取,这个异步操作的结果是一个未来的期待值,所以被称为future .

future和promise的作用是在不同线程之间传递数据。

假设线程1需要线程2的数据,那么组合使用方式如下:

  • 线程1初始化一个promise对象和一个future对象,promise传递给线程2,相当于线程2对线程1的一个承诺;future相当于一个接受一个承诺,用来获取未来线程2传递的值

  • 线程2获取到promise后,需要对这个promise传递有关的数据,之后线程1的future就可以获取数据了。

  • 如果线程1想要获取数据,而线程2未给出数据,则线程1阻塞,直到线程2的数据到达

thread9


thread10

future与package_task的使用

std::packaged_task包装一个可调用的对象,并且允许异步获取该可调用对象产生的结果。

std::packaged_task将其包装的可调用对象的执行结果传递给一个std::future对象,与std::promise某种程度上是很像的,promise保存一个共享状态的值,而package_task保存的是一个函数。


thread11

线程异步操作函数async的用法

std::async比std::packaged_task,std::promise中,std::thread更高一层,它可以直接用来创建异步的task,异步的结果也保存在future中。完成后,外面再通过future.get/wait来获取这个未来的结果,强烈推荐使用async,我们不需要关注异步任务的结果,只要等待任务完成获取值就行了。

现在来看看std::async的原型async(std::launch::async | std::launch::deferred, f, args...),第一个参数是线程的创建策略,有两种策略,默认的策略是立即创建线程:

  • std::launch::async 在调用async就开始创建线程。
  • std::launch::deferred 延迟加载方式创建线程。调用async时不创建线程,直到调用了future的get或者wait时才创建线程。

第二个参数是线程函数,第三个参数是线程函数的参数。


thread12

std::future::wait_for()函数作用

template< class Rep, class Period >
std::future_status wait_for( const std::chrono::duration<Rep,Period>& timeout_duration ) const;

等待结果变得可用。阻塞直至经过指定的 timeout_duration ,或结果变为可用,两者的先到来者。返回值鉴别结果的状态。

此函数可能由于调度或资源争议延迟而阻塞长于 timeout_duration 。

推荐标准库用稳定时钟度量时长。若实现用系统时钟代替,则等待时间可能也对时钟调整敏感。

若调用此函数前 valid()== false 则行为未定义。

参数 解释
future_status::deferred 要计算结果的函数仍未启动
future_status::ready 结果就绪
future_status::timeout 已经过时限

异常

  • 时钟、时间点或时长在执行中可能抛的任何异常(标准库提供的时钟、时间点和时长决不抛出)。

注意

  • 鼓励实现在调用前检测 valid == false 的情况并抛出以 future_errc::no_state 为 error_condition 的 future_error .

Re:

https://www.cnblogs.com/DOMLX/p/10945309.html