C++线程操作

本文详细介绍了C++中的线程概念及使用,包括线程状态、线程创建、join()与detach()、互斥量mutex和条件变量condition_variable的应用。通过实例展示了如何在多线程环境中同步和通信,以及如何获取线程返回值。强调了线程安全和资源管理的重要性。
摘要由CSDN通过智能技术生成


一、线程和进程的基本概念

 
线程是操作系统进行运算调度的基本单位,进程是操作系统进行资源分配的基本单位。
不同进程分配不同的内存区域,同一个进程中的不同线程共享同一片内存。
 
线程的不同状态:

  1. 新建状态:仅仅是语言层面创建了线程状态,但是还没有与操作系统线程进行关联;
  2. 就绪状态:线程已经与操作系统相关联,可以被CPU进行调度;
  3. 运行状态:线程正占用着CPU,正在运算;
  4. 阻塞状态:等待被唤醒的状态;
  5. 终止状态:线程运行结束。

并行(Parallel)是真正意义上的同时运行,在同一时间点有多个程序在同时运行,比如多核CPU多个核同时运行着不同的程序。并发(ConCurrent)是在同一个时间段上多个程序同时运行,在宏观上看是同时运行,微观上是交替运行。
 
之所以设计多线程是为了充分的利用CPU资源,比如线程A正在等待磁盘数据的时候,可以将CPU让出来,线程B就能占用CPU进行计算。
 

二、C++中的线程

 
c++11新增了与操作系统无关的线程类thread,使用方法如下:

void func(int i) {
    cout << i << endl;
}

int main() {
    thread t1(func, 10);  //传入线程绑定的函数和函数所需参数
    t1.join();
    return 0;
}

thread对象创建之后,线程就已经进入就绪态,可以占用CPU进行运算。
线程的运行需要以具体的函数作为入口,而函数的参数传递无非值传递,引用传递和指针传递3种形式,值传递和指针传递没什么好说的,直接复制值或者指针放入线程入口函数中,而线程的使用中需要特别注意⚠️⚠️⚠️引用传递,普通函数使用引用传递,底层使用的还是指针,所以函数中的引用会关联到函数外的变量。但是在线程的使用中,函数引用传递是引用的复制,函数中的引用和函数外的值没有关联关系。如果想关联,必须传递参数的时候使用std::ref()语义进行包裹。
 
join()detach()的用法:调用线程的join()方法意味着,主线程会阻塞在该语句处,等待子线程运行结束的时候才会继续运行主线程,而detach()方法意味着,子线程从主线程中分离出来各自运行,子线程变为后台守护线程(deamon thread),如此这般,子线程可能会结束在主线程之后,可能会出现不可预知的问题。
 

三、互斥量mutex的使用

 
当存在多个线程共同访问同一资源(比如队列)的时候,如果操作的顺序不当可能会出现不可预知的错误,该资源成为临界资源,此时C++使用互斥量mutex进行资源访问的控制。
 
通常为了方便理解多线程操作,我们将mutex称为锁🔒,对资源的访问称为对资源上锁和解锁。实际上mutex是一个标记量,不同的线程对同一mutex对象进行lock()unlock()可以达到控制资源访问次序的目的。
 
lock()方法是尝试获取锁并加锁,如果获取不到锁,线程就会阻塞在此语句处,mutex调用了lock()就必须unlock(),否则就有可能是其他线程不能成功lock()而一直阻塞。
为了解决这种可能忘记unlock()的问题,引入lock_guard模版类,lock_guard类似于智能指针,出作用域会自动unlock()。和lock_guard相似的一个模版类是unique_lock,使用方法和lock_guard类似。
 

四、条件变量condition_variable的使用

 
多个线程需要通信的时候会用到cv,比如消费者和生产者线程模型,这两个线程并不是单单的对资源解锁和加锁。而是需要通信,比如,消费者没有资源可以消费的时候需要将生产者叫醒。
 
cv有两个方法wait()wait_for()wait方法会让当前线程释放锁然后阻塞,直到被唤醒;wait_for方法需要传递一个时间参数,当前线程释放锁然后等待被唤醒,但是不会一直等待,而是吵过时间参数自动苏醒。
cv还有两个方法notify_one()notify_all()notify_one会唤醒线程等待队列的第一个线程,而notify_all会唤醒所有等待的线程。
 
waitsleep方法的区别:

  1. waitcv的方法,sleepthis_thread的方法;
  2. wait会释放锁然后再阻塞,而sleep直接阻塞,不会释放锁。

 

五、如何获得线程的返回值

 
前面提到的线程入口函数都是实现一系列的操作但是没有返回值,如果需要线程的返回值,我们需要使用std::future<>这样一个模版类来接受返回值,而普通的用thread创建线程的方式不能获得future对象,此时有数种方法:

  1. 使用异步任务函数std::async(),顾名思义,异步任务就是启动一个和主线程并发的任务,该方法会返回一个future对象,通过该对象的get()方法就能获得线程返回的对象,使用方法如下:
int func() {
    return 1234;
}

int main() {
    std::future<int> fval = std::async(func);
    std::cout << fval.get() << std::endl;
    return 0;
}

主线程会阻塞在get()处,直到线程返回返回值。async()方法还可以传入一个枚举标志量std::launch,如果将此标记量设置为deferred,意味着方法调用处不会立即创建线程,直到调用get()方法才会创建方法并运行,而此时和直接在主线程调用线程入口函数是一样的,是串行运行。注意⚠️:get()方法只能调用一次;
 
2. 使用std::package_task<>包裹一个函数,然后传递给一个线程对象,线程运行起来之后用package_task对象的get_future()方法获得future对象,然后再用get()方法拿到返回值。package_task使用方法如下:

int func() {
    return 1234;
}

int main() {
    std::packaged_task<int()> task(func);
    std::thread t1(std::ref(task));
    t1.join();
    std::cout << task.get_future().get() << std::endl;
    return 0;
}

 

  1. 使用std::promise<>模版类可以从线程中将值带到线程外,promise对象会传递到线程入口函数中去,然后在线程运行过程中,使用set_value()方法给promise对象赋值,然后在异步线程之外,使用promise对象的get_future()方法获得future对象,之后的操作和以上就一样了。使用方法如下:
void func(std::promise<int> &pro) {
    pro.set_value(100);
}

int main() {
    std::promise<int> pro;
    std::thread t1(func, std::ref(pro));
    std::cout << pro.get_future().get() << std::endl;
    t1.join();  //此处使用join其实意义不大,因为get已经获取到值说明异步线程已经结束
    return 0;
}

 

六、线程的其他使用

  1. wait被唤醒之后第一件事是会反复尝试直到拿到锁再往下执行,wait被唤醒拿到锁之后一般还要再次检查阻塞条件保证多线程共享的正确性;
  2. unique_lock所有权的转移类似于unique_ptr,使用move进行所有权转移;
  3. std::lock(mutex1, mutex2)方法可以尝试同时锁住多个互斥量,如果有一个没锁成功,就不会对任何互斥量进行加锁;
  4. release()方法将unique_lockmutex解绑,并返回之前绑定的mutex
  5. 互斥量mutexlock()unlock()之间包裹的语句多少称为锁的粒度,粒度要合适,如果太小可能不能达到上锁的预期,如果太大,会降低多线程运行效率;
  6. std::adopt_lock可以传递给lock_guardunique_lock的构造函数,表示这个互斥量已经被lock了(你必须要把互斥量提前lock了 ,否者会报异常);std::adopt_lock标记的效果就是假设调用一方已经拥有了互斥量的所有权(已经lock成功了);通知lock_guard不需要再构造函数中lock这个互斥量了。unique_lock也可以带std::adopt_lock标记,含义相同,就是不希望再unique_lock的构造函数中lock这个mutex。用std::adopt_lock的前提是,自己需要先把mutex lock上;用法与lock_guard相同;
  7. std::try_to_lock会尝试用mutex的lock去锁定这个mutex,但如果没有锁定成功,也会立即返回,并不会阻塞在那里,用这个try_to_lock的前提是你自己不能先lock;
  8. std::defer_lock的前提是,你不能自己先lock,否则会报异常,std::defer_lock的意思就是并没有给mutex加锁,初始化了一个没有加锁的mutex
  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
线程操作Vector可以提高程序的运行效率和并发性。Vector是一个线程安全的容器,可以被多个线程同时访问和修改。但是在多线程操作Vector时,需要注意以下几点。 首先,多线程操作Vector需要进行同步控制,以避免数据竞争和并发访问的问题。可以通过使用互斥锁(mutex)来实现线程间的互斥操作,保证在一个线程访问Vector时,其他线程不能同时进行写操作。 其次,应该注意避免在迭代器遍历Vector的同时进行修改操作,可能会导致迭代器失效或产生未定义的行为。一种解决方案是使用智能指针,可以在遍历时保持Vector的引用计数,防止迭代器失效。 另外,多线程操作Vector还可以通过分割Vector的部分来实现并行处理,提高程序的执行效率。可以将Vector分割为多个子Vector,每个线程处理一个子Vector的数据,最后再将结果合并。 在使用线程操作Vector时,还需注意线程间的数据同步和线程安全的问题。可以使用条件变量(condition variable)来实现线程间的通信和同步。同时,应考虑使用原子操作(atomic operation)来保证对Vector的操作是原子的,从而避免数据不一致的问题。 总之,多线程操作Vector可以提高程序的性能和并发性,但同时也需要谨慎处理同步控制、迭代器失效和数据同步等问题,以确保线程安全和程序的正确性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值