当前位置: 首页 > article >正文

[项目][WebServer][TcpServer]详细讲解

  • 单例模式的线程安全
    • 需要双重判空指针,降低锁冲突的概率,提高性能
    • 原因1:
      • 当第一次实例化单例时,可能有多个线程同时到来,并且svr指针为空
      • 这时他们就会去竞争锁,但只有一个线程会最快拿到锁,并且成功实例化出单例对象
      • 但此时如果不加双重判空指针,那些也进了第一层if判断的,仍然会去实例化出对象
    • 原因2:
      • 为了线程安全,必然要加锁,加锁之后再去判空
      • 但每次调用GetInstance()都需要去获得锁,释放锁,效率低下
      • 此时再加一层外层if判空,这样就会避免后续调用GetInstance()时没必要的锁竞争
static const uint16_t PORT = 8090;
static const int BACKLOG = 128;

// 单例 -- 饿汉模式
class TcpServer
{
public:
    static TcpServer* GetInstance(uint16_t port = PORT)
    {
        static pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
        if(svr == nullptr) // 双重判空指针,降低锁冲突的概率,提高性能
        {
            // 注意线程安全
            pthread_mutex_lock(&lock);
            if(svr == nullptr)
            {
                svr = new TcpServer(port);
                svr->Init();
            }
            pthread_mutex_unlock(&lock);
        }

        return svr;
    }

    void Init()
    {
        Socket();
        Bind();
        Listen();
        LOG(INFO, "TcpServer Init ... Success");
    }

    void Socket()
    {
        _listenSock = socket(AF_INET, SOCK_STREAM, 0);
        if(_listenSock < 0)
        {
            LOG(FATAL, "Socket Error");
            exit(1);
        }

        // 设置端口复用
        int opt = 1;
        setsockopt(_listenSock, SOL_SOCKET, SO_REUSEPORT, &opt, sizeof(opt));
        LOG(INFO, "Create Listen Socket ... Success");
    }

    void Bind()
    {
        struct sockaddr_in local;
        memset(&local, 0, sizeof(local));
        local.sin_family = AF_INET;
        local.sin_addr.s_addr = INADDR_ANY;
        local.sin_port = htons(_port);

        if(bind(_listenSock, (struct sockaddr*)&local, sizeof(local)) < 0)
        {
            LOG(FATAL, "Bind Error");
            exit(2);
        }
        LOG(INFO, "Bind Socket ... Success");
    }

    void Listen()
    {
        if(listen(_listenSock, BACKLOG) < 0)
        {
            LOG(FATAL, "Listen Error");
            exit(3);
        }
        LOG(INFO, "Listen Socket ... Success");
    }

    int Sock()
    {
        return _listenSock;
    }

    ~TcpServer()
    {
        if(_listenSock >= 0)
        {
            close(_listenSock);
        }
    }
private:
    TcpServer(uint16_t port)
        : _port(port)
        , _listenSock(-1)
    {}

    TcpServer(const TcpServer&) = delete;
private:
    uint16_t _port;
    int _listenSock;
    static TcpServer* svr;
};

TcpServer* TcpServer::svr = nullptr;

http://www.kler.cn/a/300738.html

相关文章:

  • K8S开启/关闭审计日志
  • redis 分布式重入锁
  • JVM直击重点
  • 关于vite+vue3+ts项目中env.d.ts 文件详解
  • eBay账号安全攻略:巧妙应对风险
  • C语言结构体漫谈:从平凡中见不平凡
  • [计算机网络]-计网学习笔记-计网知识点总结(附完整笔记)
  • C++自学笔记35(文件操作)
  • 抖音视频素材哪里来的?抖音视频素材库在哪里找分享
  • Vue 常用语法
  • 【springboot】简易模块化开发项目整合MyBatis-plus
  • [图解]建模实例-14是哪边的流程有问题
  • 代码随想录算法训练营day32
  • 4WRA6E07-2X/G24N9K4/VL配套HE-SP2比例放大器
  • 53 - I. 在排序数组中查找数字 I
  • Trinamic医疗成功的事例之TMCM611
  • QSoundEffect 用于播放一些单调简单的声音
  • Python画笔案例-045 绘制渐变圆盘
  • 国产视频转换HDMI1.4转单/双MIPI DSI/CSI LT6911C芯片方案,带音频输出,QFN64封装 Lontium
  • JDBC连接数据库
  • git版本问题Your branch is behind ‘origin/dev‘by 2 commits,
  • C语言中的磁盘映射与共享内存详解
  • C++设计模式——State状态模式
  • 基于开源链动 2 + 1 模式、AI 智能名片与 S2B2C 商城小程序的用户忠诚度计划
  • C# UDP与TCP点发【速发速断】模式
  • HTML5中`<area>`标签深入全面解析