C++11多线程编程-创新互联

因为之前有学习过c11的并发库,最近在搞项目准备复习,本节开始就重温一下这块内容打算连着写上几篇博客去记录一下.. 

创新互联公司是一家专业提供惠济企业网站建设,专注与网站制作、成都做网站H5场景定制、小程序制作等业务。10年已为惠济众多企业、政府机构等服务。创新互联专业网站设计公司优惠进行中。

题外话get几个概念

1.进程是资源分配的基本单位,线程是调度的基本单位,注意基本二字,这并不意味着进程不能调度,进程当然可以调度;线程是基本的调度单位,只要达到线程水平就可以被调度了

2.目前据我所知哈还没有能直接写出由线程开始的程序,为啥呢?因为此时没有系统资源让你去动,跑程序就要代码,进程是资源分配的基本单位,说明你线程要依附于进程,单单创建个线程,他是没有用户资源区的,就没有代码段数据区这些,不过创建的时候线程他是一直有自己的内核栈的且独立;

3.独有堆栈才能保证自己独立运行,这里的堆栈是共享了父进程的地址空间之后在里面分配了自己独立的堆栈,就是用户态的堆栈吧,还是得依靠父进程,这也能说明了第二个问题,只有依靠了进程,有了用户资源才能跑代码不是

目录

今天先从thread线程的使用开始:

构造函数:

一些公共的api

get_id()

join()

detach()

别的几个函数

joinable()

hardware_concurrency 

线程的命名空间 std::this_thread

sleep_for()

sleep_until()

yield()

说几个小点

线程独有自己的堆栈

类的成员函数的线程化:


在学习linux的时候肯定都接触过c语言里线程pthread,不过他并不好用,各种回调函数,句柄很多参数等等,相关的api也是一大堆,不过在c++11之后就提供了线程类叫做 std::thread,用起来就很简单了。

说到线程,肯定就要考虑线程同步,线程安全的问题,同linux上一样,c++11也提供了pv原子操作,互斥(各种锁),条件变量,信号量,future(用来获取异步任务)等等这些去支持多线程的并发,后面陆续去说这些类

今天先从thread线程的使用开始: 构造函数:
// 
thread() noexcept;
默认构造函,构造一个线程对象,在这个线程中不执行任何处理动作
// 
thread( thread&& other ) noexcept;//只能移动构造了
移动构造函数,将 other 的线程所有权转移给新的 thread 对象。之后 other 不再表示执行线程。
// 
template< class Function, class... Args >explicit thread( Function&& f, Args&&... args );//可变参数模板
创建线程对象,并在该线程中执行函数 f 中的业务逻辑,args 是要传递给函数 f 的参数
任务函数 f 的可选类型有很多,具体如下:

普通函数,类成员函数,匿名函数,仿函数(这些都是可调用对象类型)
可以是可调用对象包装器类型,也可以是使用绑定器绑定之后得到的类型(仿函数)
// 
thread( const thread& ) = delete;  //不能拷贝构造
使用 =delete 显示删除拷贝构造,不允许线程对象之间的拷贝

线程中的资源是不能被复制的,因此通过 = 操作符进行赋值操作最终并不会得到两个完全相同的对象。

// move (1)	      //不允许赋值
thread& operator= (thread&& other) noexcept;
// copy [deleted] (2)	
thread& operator= (const other&) = delete;

如果 other 是一个右值,会进行资源所有权的转移
如果 other 不是右值,禁止拷贝,该函数被显示删除(=delete),不可用

一些公共的api
get_id()

返回线程的id

每个被创建出的线程实例都对应一个线程 ID,这个 ID 是唯一的,可以通过这个 ID 来区分和识别各个已经存在的线程实例

函数原型
std::thread::id get_id() const noexcept;

回收线程资源:有两种

join()和detach()

用的时候只能二选一

join()

join() 字面意思是连接一个线程,意味着主动地等待线程的终止(线程阻塞)。在某个线程中通过子线程对象调用 join() 函数,调用这个函数的线程被阻塞,但是子线程对象中的任务函数会继续执行,当任务执行完毕之后 join() 会清理当前子线程中的相关资源然后返回,同时,调用该函数的线程解除阻塞继续向下执行。

注意:join在哪个线程里面调用,哪个线程走到join就会阻塞这个线程,比如在main这个主线程去创建了一个子线程funa,funa执行,main也走他的,当走到funa的join时候,main就阻塞在这里了,等着funa走完,返回回来之后main才继续走

很简单的道理

函数原型
void join();
detach()

detach() 函数的作用是进行线程分离,分离主线程和创建出的子线程。在线程分离之后,主线程退出也会一并销毁创建出的所有子线程,在主线程退出之前,子线程可以脱离主线程继续独立的运行,任务执行完毕之后,这个子线程会自动释放自己占用的系统资源

函数原型
void detach();

用了这个函数就好像,孩子叛逆了离家出走跑到后台去了,父母管不着了,父线程管不着子线程了,他们各干各的

对于join和detach个人感觉,用join好一点,毕竟join了父线程必须要等到子线程的结果才行,而detach的话,你也不知道你父线程运行完了,子线程有没有走完,没有走完如果父线程结束程序结束,那结果肯定是由问题的不是~~当然了这是个人见解,学艺不精

给出示例,把上面我说的都演示了一下。

void* funa(void*)
{
	cout<< "void *"<< endl;
	return nullptr;
}
void funb()
{
	cout<< "void"<< endl;
	return;
}
void func(int a, int b)
{
	int c = a + b;
	cout<< "func"<< endl;
	cout<< "c= "<< c<< endl;
}
void dosomething()
{
	cout<< "我要干活了"<< endl;
	cout<< "hhhhhhhh"<< endl;
	cout<< "mainmainmain"<< endl;
}
int main()
{
	//创建线程对象
	thread tha(funa, nullptr);
	thread thb(funb);
	thread thc(func, 10, 20);

	cout<< "funa的线程id:"<< tha.get_id()<< endl;
	cout<< "funb的线程id:"<< thb.get_id()<< endl;
	tha.join();//主线程要等待创建的这些线程结束
	thb.join();
	thc.join();

	//main解除阻塞了,才往下走
	dosomething();
	return 0;
}

把join都换成detach,就会发现结果很多次有的子线程的结果都没有完全打印,就是因为父线程结束了,二者分离子线程还没完,除非在父线程结束前你加个sleep休眠一下,等等子线程。像linux下不管是进程fork了或者线程,刚开始学习的时候sleep也是挺常见的操作,毕竟都是些操作系统底层的问题

还有就是你会发现每次打印的结果,经常会乱序,这就是因为线程嘛,不加以控制,大家都在一起走,谁先走完就不确定了。我开篇也说了线程同步,安全这是多线程并发必须要控制的,用互斥,条件变量这些去控制,后面我会分别讲的。。

这两句话算我多啰嗦,能点进来看我这个博客的肯定对于进程和线程的问题,怎么控制这些应该是清楚的。

别的几个函数  joinable()

joinable() 函数用于判断主线程和子线程是否处理关联(连接)状态,一般情况下,二者之间的关系处于关联状态,该函数返回一个布尔类型:

返回值为 true:主线程和子线程之间有关联(连接)关系
返回值为 false:主线程和子线程之间没有关联(连接)关系

bool joinable() const noexcept;
void foo()
{
    this_thread::sleep_for(std::chrono::seconds(1));
}

int main()
{
    thread t;
    cout<< "before starting, joinable: "<< t.joinable()<< endl;

    t = thread(foo);
    cout<< "after starting, joinable: "<< t.joinable()<< endl;

    t.join();
    cout<< "after joining, joinable: "<< t.joinable()<< endl;

    thread t1(foo);
    cout<< "after starting, joinable: "<< t1.joinable()<< endl;
    t1.detach();
    cout<< "after detaching, joinable: "<< t1.joinable()<< endl;
}

  • 在创建的子线程对象的时候,如果没有指定任务函数,那么子线程不会启动,主线程和这个子线程也不会进行连接
  • 在创建的子线程对象的时候,如果指定了任务函数,子线程启动并执行任务
  • 主线程和这个子线程自动连接成功在主线程调用了join()函数,子线程中的任务函数继续执行,直到任务处理完毕,这时join()会清理(回收)当前子线程的相关资源,所以这个子线程和主线程的连接也就断开了,因此,调用join()之后再调用joinable()会返回false。
  • 子线程调用了detach()函数之后,父子线程分离,同时二者的连接断开,调用joinable()返回false
hardware_concurrency 

thread 线程类还提供了一个静态方法,用于获取当前计算机的 CPU 核心数(是虚拟核),根据这个结果在程序中创建出数量相等的线程,每个线程独自占有一个CPU核心,这些线程就不用分时复用CPU时间片,此时程序的并发效率是最高的。

static unsigned hardware_concurrency() noexcept;

实际上我的电脑本身是4核处理,这个函数返回的其实是虚拟的cpu核心数,也就是逻辑处理器。我这个电脑物理处理器是4,逻辑处理器是8

int main()
{
    int num = thread::hardware_concurrency();
    cout<< "CPU number: "<< num<< endl;
}

线程的命名空间 std::this_thread

在这个命名空间中提供了四个公共的成员函数,通过这些成员函数就可以对当前线程进行相关的操作了

get_id()前面说了,说说剩下三个

进程一共有五种状态分别为:创建态,就绪态,运行态,阻塞态(挂起态),退出态(终止态) 其中创建态和退出态维持的时间是非常短的,稍纵即逝。我们主要是清楚就绪态 , 运行态 , 挂起态

线程创建后同样有这五个状态

sleep_for()

线程和进程的执行有很多相似之处,在计算机中启动的多个线程都需要占用 CPU 资源,但是 CPU 的个数是有限的并且每个 CPU 在同一时间点不能同时处理多个任务。

为了能够实现并发处理,多个线程都是分时复用CPU时间片,快速的交替处理各个线程中的任务。因此多个线程之间需要争抢CPU时间片,抢到了就执行,抢不到则无法执行(因为默认所有的线程优先级都相同,内核也会从中调度,不会出现某个线程永远抢不到 CPU 时间片的情况)。

this_thread 中提供了一个休眠函数 sleep_for(),调用这个函数的线程会马上从运行态变成阻塞态并在这种状态下休眠一定的时长,因为阻塞态的线程已经让出了 CPU 资源,代码也不会被执行,所以线程休眠过程中对 CPU 来说没有任何负担。

这个函数是函数原型如下,参数需要指定一个休眠时长,是一个时间段:

templatevoid sleep_for (const chrono::duration& rel_time);

this_thread::sleep_for(chrono::seconds(1));//秒
this_thread::sleep_for(chrono::milliseconds(1000));//毫秒
this_thread::sleep_for(chrono::microseconds(1000*1000));//微妙

注意:程序休眠完成之后,会从阻塞态重新变成就绪态,就绪态的线程需要再次争抢 CPU 时间片,抢到之后才会变成运行态,这时候程序才会继续向下运行。 

sleep_until()

sleep_until():指定线程阻塞到某一个指定的时间点 time_point类型,之后解除阻塞
sleep_for():指定线程阻塞一定的时间长度 duration 类型,之后解除阻塞

templatevoid sleep_until (const chrono::time_point& abs_time);

sleep_until() 和 sleep_for() 函数的功能是一样的,参数不同,实际开发具体选择

对于这两个函数学习之前要看看C++11 中提供了日期和时间相关的库 chrono,chrono 库主要包含三种类型的类:时间间隔duration、时钟clocks、时间点time point。也就是这俩函数要用到的参数

yield()

在线程中调用这个函数之后,处于运行态的线程会主动让出自己已经抢到的 CPU 时间片,最终变为就绪态,这样其它的线程就有更大的概率能够抢到 CPU 时间片了。使用这个函数的时候需要注意一点,线程调用了 yield () 之后会主动放弃 CPU 资源,但是这个变为就绪态的线程会马上参与到下一轮 CPU 的抢夺战中,不排除它能继续抢到 CPU 时间片的情况

std::this_thread::yield() 的目的是避免一个线程长时间占用CPU资源,从而导致多线程处理性能下降
std::this_thread::yield() 是让当前线程主动放弃了当前自己抢到的CPU资源,但是在下一轮还会继续抢

void yield() noexcept;
说几个小点 线程独有自己的堆栈

多线程调用同一个方法,局部变量会共享吗_

//线程函数的返回值是没有意义的,不要取得线程的返回值
void funa(char c)
{
	cout<< c<< endl;
	int x = 10;
	cout<< &x<< endl;
	return;
}
int main()
{
	thread tha(funa,'a');
	thread thb(funa,'b');
	//每个线程独有自己的栈
	//线程a和b 不共享x这个局部变量,x打印不一样,因为x在不同的系统调用栈中
	tha.join();
	thb.join();
	return 0;
}

局部变量是线程安全的【高并发】终于弄懂为什么局部变量是线程安全的了!! 

类的成员函数的线程化:
class Test {
	int value;
public:
	Test() {}
	~Test(){}
	static void funa(int a)//没this
	{
		cout<< "static funa=="<< a<< endl;
	}
	//this ,a
	void setValue(const int x)
	{//void setValue(Test*const this,const int x)
		
		value = x;
		cout<< "setVaule:"<< value<< endl;
	}
};
int main()
{
	thread tha(Test::funa, 10);//不是全局,静态要指明
	tha.join();

	int b = 20;
	Test t;
	thread thb(&Test::setValue,&t,b);//有this 要指明对象地址
	thb.join();
	return 0;
}
class AA {
public:
	void operator()(int a, int b)//仿函数
	{
		int c = a + b;
		cout<< "c;"<< c<< endl;
	}
	void mul(int a, int b)//普通的成员函数
	{
		int d = a * b;
		cout<< "d; "<< d<< endl;
	}
};
int main()
{
	thread tha(AA(), 10, 20);//仿函数当做可调用对象
	tha.join();
	AA aa;
	thread thb(&AA::mul, &aa, 5, 3);
	thb.join();

    int c = 20;//lambda表达式,也可以当线程化的可调动对象
	thread thc([&](int x)->void {c += x; }, 50);
	cout<< "c:"<< c<< endl;
	thc.join();
	return 0;
}
class Test {
public:
	int value;
	void func(){//this
    cout<<"func"<

ok完事收工。

补充

有关__thread和thread_local的解释

__thread是GCC内置的线程局部存储设施,存取效率可以和全局变量相比。__thread变量每一个线程有一份独立实体,各个线程的值互不干扰。

被thread修饰的变量,在多线程中不是共享的而是每个线程单独一个,个人理解就是同一个变量即使多个线程同时调用,在同样的动作下也应该出现同样的一个结果。 既然一份变量,那就用static声明一下,验证即使static依旧可以满足前面的论述。 

写一段示例代码自己跑一下

thread_local int i = 0;
void* funa(void *ll)
{
	i += 1;
	cout<< "ai:"<< i<< " "<< &i<< endl;
	return nullptr;
}
void* funb(void* ll)
{
	i += 1;
	cout<< "bi:"<< i<< " "<< &i<< endl;
	return nullptr;
}
int main()
{
	i += 3;
	thread aa(funa,nullptr);
	aa.join();	
	thread bb(funb, nullptr);
	bb.join();//打出来两个i地址一样,因为aa.join之后资源释放把栈归还之后,bb有可能会拿到这个栈空间
	
	cout<< "main:"<< i<< " "<< &i<< endl;
	return 0;
}

有关__thread资料很少,找到一个不错的 thread_local与__thread的区别_

你是否还在寻找稳定的海外服务器提供商?创新互联www.cdcxhl.cn海外机房具备T级流量清洗系统配攻击溯源,准确流量调度确保服务器高可用性,企业级服务器适合批量采购,新人活动首月15元起,快前往官网查看详情吧


新闻名称:C++11多线程编程-创新互联
当前链接:http://pcwzsj.com/article/doigdj.html