C++ 單例模式 call_once : terminate called after throwing an instance of ‘std::system_error‘

在學習了C++中可以使用call_once進行初始化資源后,我就想著寫一個單例模板供以后使用。

template<typename T>
class SingleTon {using Ptr = std::shared_ptr<T>;static Ptr p;static std::once_flag flag;template<typename ...Args>static void init(Args&&...args) {p.reset(new T(std::forward<Args>(args)...));}public:template<typename ...Args>static Ptr getInstance(Args&& ...args) {/*//也可以使用lambda表達式實現上面成員函數的功能,但是不是很必要,使用成員函數更加直觀而且避免了每次調用都創建lambda表達式和傳遞p的性能消耗auto init = [](Ptr &p, auto&& ...args1) {p.reset(new T(std::forward<decltype(args1)>(args1)...));};*///如果不使用lambda表達式而是使用靜態成員函數進行初始化,則必須在傳入給call_once的時候就進行實例化//否則編譯器會提示init是一個未解決的重載函數std::call_once(flag, init<Args...>, std::forward<Args>(args)...);//static Ptr p = std::make_shared<T>(std::forward<Args>(args)...);return p;}
};template<typename T>
std::shared_ptr<T> SingleTon<T>::p;template<typename T>
std::once_flag SingleTon<T>::flag;

但是非常不幸,如果我們簡單測試一下這個單例模式就會報錯。

#include "Singleton.h"
#include <iostream>
#include <string>int main() {auto p = SingleTon<std::string>::getInstance("Test");std::cout << *p << std::endl;return 0;
}

報錯信息:

terminate called after throwing an instance of 'std::system_error'what():  Unknown error -1

這不禁讓我非常疑惑,為什么看起來沒有什么問題的程序會報這么嚴重的錯誤呢?思考沒有頭緒后我在google上搜索了一下,發現已經有人遇到了這個問題:https://stackoverflow.com/questions/65335620/terminate-called-after-throwing-an-instance-of-stdsystem-error

大概的原因應該是因為使用的動態鏈接庫,我們沒有使用pthread_create,所以就沒有pthread_create的定義。然而call_once又要使用pthread_create(GNU C++ standard library std::call_once checks whether the application is multi-threaded by checking whether pthread_create can be resolved),所以就導致出現了這么嚴重的錯誤。

所以解決方案就是我們必須在多線程環境下使用call_once。就我們這里來講,我們只需要讓程序里連接pthread庫就可以了。(感謝@lingwq-lingwq的糾正,我剛開始以為是必須要創建一次線程)

自己對于靜態庫、動態庫的理解還是不夠深入,應該花時間再專門學習一下。
這里我們訪問模板類的靜態成員對象,雖然在每個文件中都生成了靜態成員對象,但是在鏈接的時候鏈接器將隨機選擇一個目標中的空間作為最終存儲空間,從而實現了多個文件中的實例化類模板共享同一套靜態成員。

C++11規定對于函數局部靜態變量的初始化只會在某一線程上單獨發生,在初始化完成之前,其他線程不會越過靜態數據的聲明而繼續運行。因此我們也可以用局部靜態變量來實現線程安全的單例模式。

template<typename T>
class SingleTon {using Ptr = std::shared_ptr<T>;public:static Ptr getInstance() {static Ptr p = std::make_shared<T>();    //對于靜態變量的初始化只會進行一次return p;}
};

這種實現簡單高效,而且沒有使用call_once要求的多線程環境。因為是模板類,所以在實例化之前也不用擔心內存浪費,唯一的缺點就是不能夠在創建的時候傳入參數。

需要注意這里的getInstance函數不能是模板函數,如果是模板函數那么傳入不同的參數就會得到不同的實例化,產生不同的局部靜態對象。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/383513.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/383513.shtml
英文地址,請注明出處:http://en.pswp.cn/news/383513.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

C++讀寫鎖造成死鎖

C14支持std::shared_timed_mutex C17支持std::shared_mutex 前者相比后者支持的操作更多&#xff0c;但是后者相對性能更好。 使用std::lock_guard<std::shared_mutex>和std::unique_lock<std::shared_mutex>互斥訪問使用std::shared_lock<std::shared_mutex…

每日一題:449. 序列化和反序列化二叉搜索樹

題目分析 題目鏈接&#xff1a;449. 序列化和反序列化二叉搜索樹 覺得序列化很簡單&#xff0c;前序遍歷、后序遍歷、中序遍歷、層序遍歷等等。其中得到前序遍歷和后序遍歷是可以通過遞歸解法反序列化的&#xff0c;覺得這樣子做有點復雜。就想著可不可以一次遍歷。一次遍歷的…

C++高效集合數據結構設計

緒論 在復雜算法實現過程中我們經常會需要一個高效的集合數據結構&#xff0c;支持常數級別的增、刪、查&#xff0c;以及隨機返回、遍歷&#xff0c;最好還能夠支持交集、并集、子集操作 哈希集合實現 大家可能很快想到unordered_set&#xff0c;unordered_set由于底層是哈…

C++ 工具函數庫

在寫一些大型項目的過程中經常需要一些工具函數&#xff0c;例如獲取隨機數、計時器、打印函數、重要常量&#xff08;如最大值&#xff09;、信號與槽等&#xff0c;由于每一個工程都自己手動實現一個實在是太傻&#xff0c;我將其總結放入一個文件中。 utils.h // Copyright…

muduo網絡庫使用入門

muduo網絡庫介紹 muduo網絡庫是陳碩大神開發的基于主從Reactor模式的&#xff0c;事件驅動的高性能網絡庫。 網絡編程中有很多是事務性的工作&#xff0c;使用muduo網絡庫&#xff0c;用戶只需要填上關鍵的業務邏輯代碼&#xff0c;并將回調注冊到框架中&#xff0c;就可以實…

C++ map/unordered_map元素類型std::pair<const key_type, mapped_type>陷阱

在開發的過程中需要遍歷一個unordered_map然后把他的迭代器傳給另一個對象&#xff1a; class A; class B { public:void deal(const std::pair<int, A>& item); }; std::unordered_map<int, A> mp; B b; for (auto &pr : mp) {b.deal(pr); }在我的項目中…

Ubuntu install ‘Bash to dock‘

緒論 在Ubuntu環境搭建這篇博客中記錄了使用Dash To Dock來配置Ubuntu的菜單項&#xff0c;使得實現macOS一樣的效果。為了配置新電腦的環境&#xff0c;我還是想安裝這個軟件。但是如今在Ubuntu Software中已經找不到這個軟件了&#xff0c;我在網上借鑒了一些博客的經驗才得…

Leetcode第309場周賽

Date: September 4, 2022 Difficulty: medium Rate by others: ???? Time consuming: 1h30min 題目鏈接 競賽 - 力扣 (LeetCode) 題目解析 2399. 檢查相同字母間的距離 class Solution {public:bool checkDistances(string s, vector<int>& distance) {vec…

C++ 模板函數、模板類:如果沒有被使用就不會被實例化

C中如果一個模板函數沒有使用過&#xff0c;那么其局部靜態變量都不會被實例化&#xff1a; class A { public:A() {edward::print("A ctor");} };template<typename T> void test() {static A a; }int main() {test<int>(); //如果注釋掉則不會有輸出r…

C++ 條件變量的使用

緒論 并發編程紛繁復雜&#xff0c;其中用于線程同步的主要工具——條件變量&#xff0c;雖然精悍&#xff0c;但是要想正確靈活的運用卻并不容易。 對于條件變量的理解有三個難點&#xff1a; 為什么wait函數需要將解鎖和阻塞、喚醒和上鎖這兩對操作編程原子的&#xff1f;為…

C++Primer學習筆記:第1章 開始

本博客為閱讀《C Primer》&#xff08;第5版&#xff09;的讀書筆記 ps:剛開始的時候我將所有的筆記都放在一篇博客中&#xff0c;等看到第六章的時候發現實在是太多了&#xff0c;導致我自己都不想看&#xff0c;為了日后回顧&#xff08;不那么有心理壓力&#xff09;&#…

【ubuntu】ubuntu14.04上安裝搜狗輸入法

** 在ubuntu14.04.4 desktop 64amd版本上安裝sogou輸入法 ** 0.換安裝源為中國源&#xff08;可選&#xff0c;下載會快些&#xff09; 1.搭fcitx環境 2.安裝sogou for linux 詳細步驟&#xff1a; 因為sogou中文輸入法基于fcitx(Free Chinese Input Toy for X),需要先搭環境…

【ubuntu】ubuntu下用make編譯程序報錯找不到openssl/conf.h

ubuntu下用make編譯程序報錯找不到openssl/conf.h 安裝libssl-dev:i386&#xff0c;sudo apt-get install libssl-dev:i386 看好版本&#xff0c;如果不加i386默認下載的是32位&#xff0c;用ln命令連接過去也還是用不了的!libssl.dev安裝好后&#xff0c;用find / -name libs…

【ubuntu】ubuntu如何改變系統用戶名

ubuntu如何改變系統用戶名 方法1&#xff1a;修改現有用戶名 方法2&#xff1a;創建新用戶&#xff0c;刪掉舊用戶 方法1&#xff1a; * *—&#xff01;&#xff01;&#xff01;有博客說要先改密碼&#xff0c;再改用戶名&#xff0c;否則會出現無法登陸狀況&#xff01;&…

什么是signal(SIGCHLD, SIG_IGN)函數

什么是signal(SIGCHLD, SIG_IGN)函數 在進行網絡編程時候遇到這個函數的使用&#xff0c;自己學習結果如下&#xff0c;有不對請幫忙指正:) signal(SIGCHLD, SIG_IGN)打開manpage康一康~ sighandler_t signal ( int signum, sighandler_t handler );參數1 int signum: 就是…

ssh連接不上linux虛擬機

ssh連接不上linux虛擬機 1.開啟ssh服務 linux虛擬機下命令行輸入&#xff1a; start service ssh如果顯示沒有ssh&#xff0c;就下面兩個試一試哪一個ok&#xff0c;安裝一下ssh&#xff1a; sudo apt-get install openssh-server sudo apt-get install sshd2.還有人說可能是…

沒寫client,想先測試server端怎么辦?

沒寫client&#xff0c;想先測試server端怎么辦&#xff1f; 辦法&#xff1a; 1.先打開終端./server&#xff0c;運行起來server 2.再開一個終端&#xff0c; 輸入nc 127.0.0.1 8888 回車&#xff08;這里port號要和server里邊設置的一致&#xff0c;127.0.0.1是和本機的測試…

【報錯解決】linux網絡編程報錯storage size of ‘serv_addr’ isn’t known解決辦法

linux網絡編程報錯storage size of ‘serv_addr’ isn’t known解決辦法 報錯如下&#xff1a; server.c:18:21: error: storage size of ‘serv_addr’ isn’t known struct sockaddr_in serv_addr, clit_addr; ^server.c:18:32: error: storage size of ‘clit_addr’ isn’…

【c】寫頭文件要加#ifndef,#define, #endif

頭文件首位 編寫.h時&#xff0c; 最好加上如下&#xff0c;用來防止重復包含頭文件&#xff1a; 例如&#xff1a; 要編寫頭文件test.h 在頭文件開頭寫上兩行&#xff1a;#ifndef _TEST_H#define _TEST_H// 文件名的大寫#endif頭文件結尾寫上一行&#xff1a;#endif這樣做是為…

【c】【報錯解決】incompatible implicit declaration

【報錯解決】incompatible implicit declaration 背景; 1.自己封裝的函數wrap.c包含&#xff1a; #include "wrap.h"2.主函數調用如下&#xff1a; #include <stdio.h> #include <stdlib.h> ... #include <errno.h> #include "wrap.h"…