Post

c++11 singleton

c++11 singleton

在C++11标准中,局部静态变量的初始化是线程安全的,这就意味着在多线程环境中可以保障局部静态变量只会被初始化一次,而不会引发竞态条件问题。

因此11支持的一个比较简单的写法, 这种由编译器保证的单例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
Singleton& get() {
  static Singleton instance;
  return instance;
}
// 或者
class Singleton {
public:
  static Singleton &instance() {
    static Singleton s;
    return s;
  }

  // 删除复制构造函数和赋值操作符
  Singleton(const Singleton &) = delete;
  Singleton &operator=(const Singleton &) = delete;

protected:
  Singleton() {}
  ~Singleton() {}
};

返回指针的话

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
class Singleton {
 public:
  Singleton(Singleton const&) = delete;
  Singleton& operator=(Singleton const&) = delete;
  ~Singleton() {}
  // Set value has to be thread-safe.
  void set_value(int value)
  {
    std::lock_guard<std::mutex> lock(m_mutex);
    m_value = value;
  }
  // We don't want get value to be thread-safe in our case.
  int get_value() const { return m_value; }
  static Singleton* get_instance()
  {
    // Static local variable initialization is thread-safe
    // and will be initialized only once.
    static Singleton instance{};
    return &instance;
  }

 private:
  explicit Singleton() : m_value{0} {}
  std::mutex m_mutex;
  int m_value;
};

另外也可以依赖once_flag

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
template<class T> 
class Resource
{
  Resource<T>(const Resource<T>&) = delete;
  Resource<T>& operator=(const Resource<T>&) = delete;
  static unique_ptr<Resource<T>> m_ins;
  static once_flag m_once;
  Resource<T>() = default;

public : 
  virtual ~Resource<T>() = default;
  static Resource<T>& getInstance() {
      std::call_once(m_once, []() {
          m_ins.reset(new Resource<T>);
      });
      return *m_ins.get();
  }
};

c++自动注册工厂

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
#pragma once
#include "Message.hpp"
#include <functional>
#include <map>
#include <memory>
#include <string>

struct factory {
  template <typename T> struct register_t {
    register_t(const std::string &key) {
      std::cout << "reigister key: " << key << '\n';
      factory::get().map_.emplace(key, &register_t<T>::create);
    }

    template <typename... Args>
    register_t(const std::string &key, Args... args) {
      std::cout << "reigister key: " << key << '\n';
      factory::get().map_.emplace(key, [=] { return new T(args...); });
    }
    inline static Message *create() { return new T; }
  };

  inline Message *produce(const std::string &key) {
    if (map_.find(key) == map_.end()) {
      return nullptr;
    }

    return map_[key]();
  }

  std::unique_ptr<Message> produce_unique(const std::string &key) {
    return std::unique_ptr<Message>(produce(key));
  }

  std::shared_ptr<Message> produce_shared(const std::string &key) {
    return std::shared_ptr<Message>(produce(key));
  }
  typedef Message *(*FunPtr)();

  inline static factory &get() {
    static factory instance;
    std::cout << "map size: " << instance.map_.size() << '\n';
    return instance;
  }

private:
  factory(){};
  factory(const factory &) = delete;
  factory(factory &&) = delete;

  std::map<std::string, FunPtr> map_;
};

// std::map<std::string, factory::FunPtr> factory::map_;

#define REGISTER_MESSAGE_VNAME(T) reg_msg_##T##_
#define REGISTER_MESSAGE(T, key, ...)                                          \
  static factory::register_t<T> REGISTER_MESSAGE_VNAME(T)(key, ##__VA_ARGS__);

这个就是一个自动注册工厂, 通过宏定义来注册, 通过工厂来生产对象,算是单例比较高级的运用了,可以在cpp里写注册,头文件里写注册被多次include之后会重复注册到最后的map_里,有点浪费

REF

  1. The Singleton

  2. Thread-Safe Initialization of a Singleton

  3. 深入分析C++全局变量初始化

This post is licensed under CC BY 4.0 by the author.

Trending Tags