【Linux线程篇】线程安全守护者——线程互斥(互斥锁)

W...Y的主页 😊

代码仓库分享💕

 

目录

​编辑

 引言

 Linux线程互斥

进程线程间的互斥相关背景概念

互斥量mutex

互斥量的接口

互斥量实现原理探究

可重入VS线程安全

概念

常见的线程不安全的情况

常见的线程安全的情况

常见不可重入的情况

常见可重入的情况

可重入与线程安全联系

可重入与线程安全区别


 引言

我们已经学习了线程操作,知道了线程的相关系统调用。但是在线程中一定会有一些安全问题,现在我们给与一个场景:当我们给一些线程派发抢车篇的任务时,也就是说让多个线程访问同一份资源时,会怎么样呢?我们通过一个代码来模拟抢车票的过程。

首先C++11中对我们Linux中的pthread库进行了封装,我们自己模拟了一份thread库:

#ifndef __THREAD_HPP__
#define __THREAD_HPP__
#include <iostream>
#include <pthread.h>
#include <unistd.h>
#include <string.h>
#include <functional>
using namespace std;
namespace ThreadModule
{
    template <typename T>
    using func_t = function<void(T)>;

    template <typename T>
    class Thread
    {
    public:
        Thread(func_t<T> func, T &data, const string name = "none-name")
            : _func(func), _data(data), _threadname(name), _stop(true)
        {
        }
        void Excute()
        {
            _func(_data);
        }
        static void *threadtoutine(void *args)
        {
            Thread<T> *self = static_cast<Thread<T>*>(args);
            self->Excute();
            return nullptr;
        }
        bool Start()
        {
            int n = pthread_create(&_tid, nullptr, threadtoutine, this); 
            if (!n)
            {
                _stop = false;
                return true;
            }
            else
            {
                return false;
            }
        }
        void Detach()
        {
            if (!_stop)
            {
                pthread_detach(_tid);
            }
        }
        void Join()
        {
            if (!_stop)
            {
                pthread_join(_tid,nullptr);
            }
        }
        std::string name()
        {
            return _threadname;
        }
        void Stop()
        {
            _stop = true;
        }
        ~Thread()
        {
        }

    private:
        pthread_t _tid;
        string _threadname;
        T _data; 
        func_t<T> _func;
        bool _stop;
    };
}

#endif

我们使用我们自己的thread库模拟了抢车票的任务:

#include <iostream>
#include <unistd.h>
#include <vector>
#include "Thread.hpp"

using namespace std;
using namespace ThreadModule;
int g_tickets = 10000;
const int num = 4;
class ThreadData
{
public:
    ThreadData(int &tickets, const std::string &name)
        : _tickets(tickets), _name(name), _total(0)
    {
    }
    ~ThreadData()
    {
    }

public:
    int &_tickets; // 所有的线程,最后都会引用同一个全局的g_tickets
    std::string _name;
    int _total;
};
void route(ThreadData *td)
{
    while (true)
    {
        {

            if (td->_tickets > 0) // 1
            {
                usleep(1000);
                printf("%s running, get tickets: %d\n", td->_name.c_str(), td->_tickets); // 2
                td->
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

W…Y

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值