/* -*- C++ -*- */ // $Id$ // ============================================================================ // // = LIBRARY // ace // // = FILENAME // Singleton.h // // = DESCRIPTION // // = AUTHOR // Tim Harrison (harrison@cs.wustl.edu), Douglas C. Schmidt, Chris // Lahey, Rich Christy, and David Levine. // // ============================================================================ #ifndef ACE_SINGLETON_H #define ACE_SINGLETON_H #include "ace/pre.h" #include "ace/Synch.h" #if !defined (ACE_LACKS_PRAGMA_ONCE) # pragma once #endif /* ACE_LACKS_PRAGMA_ONCE */ template class ACE_Singleton : public ACE_Cleanup { // = TITLE // A Singleton Adapter uses the Adapter pattern to turn ordinary // classes into Singletons optimized with the Double-Checked // Locking optimization pattern. // // = DESCRIPTION // This implementation is a slight variation on the GoF // Singleton pattern. In particular, a single // > instance is allocated here, // not a instance. The reason for this is to allow // registration with the , so that the // Singleton can be cleaned up when the process exits. For this // scheme to work, a (static) function must be // provided. provides one so that TYPE doesn't // need to. // // If you want to make sure that only the singleton instance of // is created, and that users cannot create their own // instances of , do the following to class : // // (a) Make the constructor of private (or protected) // (b) Make Singleton a friend of // // Here is an example: // // class foo // { // friend class ACE_Singleton; // private: // foo () { cout << "foo constructed" << endl; } // ~foo () { cout << "foo destroyed" << endl; } // }; // // typedef ACE_Singleton FOO; // // NOTE: the best types to use for ACE_LOCK are // ACE_Recursive_Thread_Mutex and ACE_Null_Mutex. // ACE_Recursive_Thread_Mutex should be used in multi-threaded // programs in which it is possible for more than one thread to // access the > instance. // ACE_Null_Mutex can be used otherwise. The reason that these // types of locks are best has to do with their allocation by // the ACE_Object_Manager. Single ACE_Recursive_Thread_Mutex // and ACE_Null_Mutex instances are used for all ACE_Singleton // instantiations. However, other types of locks are allocated // per ACE_Singleton instantiation. // public: static TYPE *instance (void); // Global access point to the Singleton. virtual void cleanup (void *param = 0); // Cleanup method, used by to destroy the // . static void dump (void); // Dump the state of the object. protected: ACE_Singleton (void); // Default constructor. TYPE instance_; // Contained instance. #if !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) static ACE_Singleton *singleton_; // Pointer to the Singleton (ACE_Cleanup) instance. #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ static ACE_Singleton *&instance_i (void); // Get pointer to the Singleton instance. }; template class ACE_Unmanaged_Singleton : public ACE_Singleton { // = TITLE // Same as , except does _not_ register with // for destruction. // // = DESCRIPTION // This version of can be used if, for example, // its DLL will be unloaded before the // destroys the instance. Unlike with , the // application is responsible for explicitly destroying the // instance after it is no longer needed (if it wants to avoid // memory leaks, at least). The static member function // must be used to explicitly destroy the Singleton. // Usage is the same as for ACE_Singleton, but note that if you // you declare a friend, the friend class must still be an // *ACE_Singleton*, not an ACE_Unmanaged_Singleton. // public: static TYPE *instance (void); // Global access point to the Singleton. static void close (void); // Explicitly delete the Singleton instance. static void dump (void); // Dump the state of the object. protected: ACE_Unmanaged_Singleton (void); // Default constructor. #if !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) static ACE_Unmanaged_Singleton *singleton_; // Pointer to the Singleton (ACE_Cleanup) instance. #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ static ACE_Unmanaged_Singleton *&instance_i (void); // Get pointer to the Singleton instance. }; template class ACE_TSS_Singleton : public ACE_Cleanup { // = TITLE // This class uses the Adapter pattern to turn ordinary classes // into Thread-specific Singletons optimized with the // Double-Checked Locking optimization pattern. // // = DESCRIPTION // This implementation is another variation on the GoF Singleton // pattern. In this case, a single > instance is allocated here, not a instance. // Each call to the static method returns a Singleton // whose pointer resides in thread-specific storage. As with // , we use the so that the // Singleton can be cleaned up when the process exits. For this // scheme to work, a (static) function must be // provided. provides one so that TYPE doesn't // need to. public: static TYPE *instance (void); // Global access point to the Singleton. virtual void cleanup (void *param = 0); // Cleanup method, used by to destroy the // singleton. static void dump (void); // Dump the state of the object. protected: ACE_TSS_Singleton (void); // Default constructor. ACE_TSS_TYPE (TYPE) instance_; // Contained instance. #if !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) static ACE_TSS_Singleton *singleton_; // Pointer to the Singleton (ACE_Cleanup) instance. #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ static ACE_TSS_Singleton *&instance_i (void); // Get pointer to the TSS Singleton instance. }; template class ACE_Unmanaged_TSS_Singleton : public ACE_TSS_Singleton { // = TITLE // Same as , except does _not_ register with // for destruction. // // = DESCRIPTION // This version of can be used if, for // example, its DLL will be unloaded before the // destroys the instance. Unlike with // , the application is responsible for // explicitly destroying the instance after it is no longer // needed (if it wants to avoid memory leaks, at least). The // static member function must be used to explicitly // destroy the Singleton. // public: static TYPE *instance (void); // Global access point to the Singleton. static void close (void); // Explicitly delete the Singleton instance. static void dump (void); // Dump the state of the object. protected: ACE_Unmanaged_TSS_Singleton (void); // Default constructor. #if !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) static ACE_Unmanaged_TSS_Singleton *singleton_; // Pointer to the Singleton (ACE_Cleanup) instance. #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ static ACE_Unmanaged_TSS_Singleton *&instance_i (void); // Get pointer to the Singleton instance. }; #if defined (__ACE_INLINE__) #include "ace/Singleton.i" #endif /* __ACE_INLINE__ */ #if defined (ACE_TEMPLATES_REQUIRE_SOURCE) #include "ace/Singleton.cpp" #endif /* ACE_TEMPLATES_REQUIRE_SOURCE */ #if defined (ACE_TEMPLATES_REQUIRE_PRAGMA) #pragma implementation ("Singleton.cpp") #endif /* ACE_TEMPLATES_REQUIRE_PRAGMA */ #include "ace/post.h" #endif /* ACE_SINGLETON_H */