C++的线程

复制代码
#include<iostream>
#include<thread>
#include<unistd.h>
using namespace std;
void myrun()
{
    while(true)
    {
    cout<<"I am a thread"<<endl;
    sleep(1);
    }
}
int main()
{
    thread t(myrun);
    t.join();
    return 0;
}

如果不添加-lpthread就会报错。

用纯c++的接口创建线程时,在Linux中还是要包含线程库。

这说明c++的那些库本质就是pthread原生线程库的封装。

这样,我们学习别的语言,只需要学习接口,因为我们知道底层是一样的。

说到底,这样的设计就是为了跨平台性。

接下来了解一下线程的局部存储。

__thread可以给每个执行流一个变量。

在编译链接库时,就会给每一个变量在线程局部存储中开辟空间。

cpp 复制代码
#include<iostream>
#include<thread>
#include<unistd.h>
using namespace std;
__thread int g_val=100;
void myrun()
{
    while(g_val--)
    {
    cout<<"I am a new thread"<<"       "<<g_val<<"      "<<&g_val<<endl;
    sleep(1);
    }
}
int main()
{
    thread t(myrun);
    while(g_val--)
    {
    cout<<"I am main thread"<<"       "<<g_val<<"      "<<&g_val<<endl;
    sleep(2);
    }
    t.join();
    return 0;
}

下面是多线程的一个简单模版。

cpp 复制代码
//test.hpp
#include<iostream>
#include<thread>
#include<unistd.h>
#include<functional>
#include<vector>
using namespace std;
template<class T>
using func_t=function<void(T)>;//返回值为void,参数为T
template<class T>
class Thread
{
    public:
    Thread(func_t<T> func,const string&name,T data)
    :_tid(0)
    ,_func(func)
    ,_threadname(name)
    ,isrunning(false)
    ,_data(data)
    {
    }
    static void*ThreadRoutine(void*args)
    {
        //(void)args;//仅仅是为了防止编译器有告警
        Thread*ts=static_cast<Thread*>(args);
        ts->_func(ts->_data);
        return nullptr;
    }
    bool Start()
    {
        int n=pthread_create(&_tid,nullptr,ThreadRoutine,this);
        if(n==0)
        {
        isrunning=true;
         return true;
        }
        return false;
    }
    bool Join()
    {
        if(!isrunning) return true;
        int n=pthread_join(_tid,nullptr);
        if(n==0)
        {
            isrunning=false;
            return true;
        }
        return false;

    }
    string GetThreadName()
    {
        return _threadname;
    }
    bool IsRunning()
    {
        return isrunning;
    }
    ~Thread()
    {}
    private:
    pthread_t _tid;
    string _threadname;
    bool isrunning;
    func_t<T> _func;
    T _data;
};
cpp 复制代码
#include"test.hpp"
using namespace std;
string GetThreadName()
{
    static int number=1;
    char name[64];
    snprintf(name,sizeof name,"Thread - %d",number++);
    return name;
}
void print(int num)
{
    while(num--)
    {
    cout<<"hello world"<<num<<endl;
    sleep(1);
    }
}
int main()
{
    int num=5;
    vector<Thread<int>> Threads;
    while(num--)
    {
        Threads.push_back(Thread<int>(print,GetThreadName(),10));
    }
    for(auto &t:Threads)
    {
        cout<<t.GetThreadName()<<" is running? "<<t.IsRunning()<<endl;
    }
    for(auto&t:Threads)
    {
        t.Start();
    }
    for(auto &t:Threads)
    {
        cout<<t.GetThreadName()<<" is running? "<<t.IsRunning()<<endl;
    }
    for(auto &t:Threads)
    {
        t.Join();
    }
    // Thread t(print,GetThreadName());
    // cout<<"Is thread running?"<<t.IsRunning()<<endl;
    // t.Start();
    // cout<<"Is thread running?"<<t.IsRunning()<<endl;
    // t.Join();
    return 0;
}
相关推荐
郝学胜_神的一滴1 小时前
CMake 27:缓存变量的特性、语法、类型与实操全解
c++·cmake
kisshyshy1 小时前
从递归到迭代,一文吃透二叉树的核心知识与 JavaScript 实现
javascript·算法·代码规范
To_OC12 小时前
LC 49 字母异位词分组:想到哈希表很简单,选对 key 才是精髓
javascript·算法·leetcode
用户9385156350717 小时前
从 O(n²) 到 O(nlogn):一文读懂快速排序的“快”与“妙”
javascript·算法
To_OC18 小时前
手写快排次次翻车?别死背快排模板了,这才是面试官想听的底层逻辑
javascript·算法·排序算法
饼干哥哥19 小时前
Reddit VOC调研太慢?搭一个AI专家团队半小时洞察任何品类|以猫用饮水机为例
人工智能·算法·ai编程
地平线开发者20 小时前
Transformer模型部署之性能优化指南
算法
地平线开发者20 小时前
人在途中:从“编译失败”到“模型可落地”——CUDA 自定义算子
算法·自动驾驶
半个落月1 天前
从递归到快速排序:用 JavaScript 把分治思想讲明白
javascript·算法·面试
小月土星1 天前
JavaScript 快速排序:从 pivot、双指针到分治思想
javascript·算法·面试