首页 > 其他 > 详细

mudou源码分析之Singleton

时间:2021-05-11 00:43:56      阅读:26      评论:0      收藏:0      [点我收藏+]

源文件 \muduo\base\Singleton.h
测试文件 \muduo\base\tests\Singleton_test.cc

Singleton.h

// that can be found in the License file.
//
// Author: Shuo Chen (chenshuo at chenshuo dot com)

#ifndef MUDUO_BASE_SINGLETON_H
#define MUDUO_BASE_SINGLETON_H

#include "muduo/base/noncopyable.h"

#include <assert.h>
#include <pthread.h>
#include <stdlib.h> // atexit

namespace muduo
{

namespace detail
{

//Using SFINAE, we can detect wether a given class has a certain member function. 
// This doesn‘t detect inherited member functions!
// http://stackoverflow.com/questions/1966362/sfinae-to-check-for-inherited-member-functions
//用来判断是否有 no_destroy 函数
template<typename T>
struct has_no_destroy
{
  template <typename C> static char test(decltype(&C::no_destroy));
  template <typename C> static int32_t test(...);
  const static bool value = sizeof(test<T>(0)) == 1;
};
}  // namespace detail

template<typename T>
class Singleton : noncopyable   //不可拷贝
{
 public:
  Singleton() = delete;   //不可调用构造和析构函数
  ~Singleton() = delete;

  static T& instance()  //返回对象
  {
    //pthread_once能够保证init()该函数只调用一次
    //能够保证线程安全
    pthread_once(&ponce_, &Singleton::init); //第一次调用时创建对象,init()中创建
    assert(value_ != NULL);
    return *value_;
  }

 private:
  static void init()
  {
    value_ = new T();
    if (!detail::has_no_destroy<T>::value)
    {
      ::atexit(destroy);  //登记销毁函数,程序结束时自动调用destroy
    }
  }

  static void destroy()
  { 
    //什么是incomlete_type? sizeof T == 0
    //如class A; A *p; A是incomlete_type,仍然可以定义指针,编译时不报错,但A类型不完整
    //char T_must_be_complete_type[-1] 数组大小为-1时编译报错,能够在编译时发现错误
    typedef char T_must_be_complete_type[sizeof(T) == 0 ? -1 : 1];
    T_must_be_complete_type dummy; (void) dummy;

    delete value_;
    value_ = NULL;
  }

 private:
  static pthread_once_t ponce_;
  static T*             value_;
};

template<typename T>
pthread_once_t Singleton<T>::ponce_ = PTHREAD_ONCE_INIT;

template<typename T>
T* Singleton<T>::value_ = NULL;

}  // namespace muduo

#endif  // MUDUO_BASE_SINGLETON_H

Singleton_test.cc

#include "muduo/base/CurrentThread.h"
#include "muduo/base/Thread.h"

#include <stdio.h>

class Test : muduo::noncopyable
{
 public:
  Test()
  {
    printf("tid=%d, constructing %p\n", muduo::CurrentThread::tid(), this);
  }

  ~Test()
  {
    printf("tid=%d, destructing %p %s\n", muduo::CurrentThread::tid(), this, name_.c_str());
  }

  const muduo::string& name() const { return name_; }
  void setName(const muduo::string& n) { name_ = n; }

 private:
  muduo::string name_;
};

class TestNoDestroy : muduo::noncopyable
{
 public:
  // Tag member for Singleton<T>
  void no_destroy();

  TestNoDestroy()
  {
    printf("tid=%d, constructing TestNoDestroy %p\n", muduo::CurrentThread::tid(), this);
  }

  ~TestNoDestroy()
  {
    printf("tid=%d, destructing TestNoDestroy %p\n", muduo::CurrentThread::tid(), this);
  }
};

void threadFunc()
{
  //先打印,后改名
  printf("tid=%d, %p name=%s\n",
         muduo::CurrentThread::tid(),
         &muduo::Singleton<Test>::instance(),
         muduo::Singleton<Test>::instance().name().c_str());
  muduo::Singleton<Test>::instance().setName("only one, changed");
}

int main()
{
  muduo::Singleton<Test>::instance().setName("only one"); //获取对象,调用setName方法
  muduo::Thread t1(threadFunc); //创建线程,线程函数threadFunc
  t1.start();
  t1.join();
  //主线程打印
  printf("tid=%d, %p name=%s\n",
         muduo::CurrentThread::tid(),
         &muduo::Singleton<Test>::instance(),
         muduo::Singleton<Test>::instance().name().c_str());
  muduo::Singleton<TestNoDestroy>::instance();
  printf("with valgrind, you should see %zd-byte memory leak.\n", sizeof(TestNoDestroy));
}

/*
test output:

tid=6944, constructing 0x9474a10
tid=6945, 0x9474a10 name=only one
tid=6944, 0x9474a10 name=only one, changed
tid=6944, constructing TestNoDestroy 0x9474f10
with valgrind, you should see 1-byte memory leak.
tid=6944, destructing 0x9474a10 only one, changed

*/

mudou源码分析之Singleton

原文:https://www.cnblogs.com/Lj-ming/p/14753399.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!