C++实现类似反射模式(模板)
生活随笔
收集整理的這篇文章主要介紹了
C++实现类似反射模式(模板)
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
C++實現類似反射模式(模板)
編程技巧 2010-11-23 21:08:02 閱讀70 評論0 ??字號:大中小?訂閱
最近在編寫遙感算法工具箱的時候,工具都是在xml文件中配置好的,在工具箱上構建一棵樹根據xml配置文件,然后通過雙擊不同的樹節點,彈出不同的算法對話框。最簡單的方式就是使用if else 或者switch case之類的條件判斷語句來實現,但是這個太不方便了,每增加一個算法,都要在分支上添加一個條件判斷,用現在流行的話說就是太不給力了。于是想通過一個比較通用的方式來解決這個問題。 由于我的算法對話框都是基于MFC的CDialog,所以可以通過算法對話框的類名來創建各自的對象,然后將對話框顯示。以前知道在Java和C#中有一個反射模式,就是可以通過類名來創建一個類的對象。但是C++是沒有這個東東的,于是就查找資料,終于用C++的模板實現了類似的功能。下面貼代碼:/*************************************************************************** * * Time: 2010-11-23 * Project: 遙感平臺? * Purpose: 圖像數據處理對話框反射模式 * Author: ?李民錄 * Copyright (c) 2010, LT_IMAGE * ****************************************************************************/ /** /file RegistClassName.h? * 圖像數據處理對話框反射模式 */
#ifndef REGISTCLASSNAME_H #define REGISTCLASSNAME_H
#include <afxwin.h> #include <map> #include <string> using namespace std;
#ifdef _DEBUG #include <assert.h> #endif
/** * @brief 重定義對象 */ typedef CDialog*(*pf)();
/** * /class ClassMap RegistClassName.h? * @brief 類映射 */ class ClassMap { private: /** * @brief 私有構造函數 * 為了讓這個類不產生實例對象 */ ClassMap() { }
/** * @brief 以類名為鍵的一個存放著構建方法的Hash表 */ static map<string, pf> m_ClassMap; public: /** * @brief 定義一個類必須注冊一下 * @param _className 類名 * @param _createFun 注冊函數指針 */ static void RegistClass(string _className, pf _createFun) { if(m_ClassMap.find(_className) != m_ClassMap.end()) return; //已經注冊過,直接返回
m_ClassMap[_className] = _createFun; }
/** * @brief 私有構造函數 * @param _className 類名 */ static CDialog* forName(string _className) { if(m_ClassMap.find(_className) == m_ClassMap.end()) { #ifdef _DEBUG assert(0); #endif return NULL; //沒有找到類對象,返回NULL }
return (m_ClassMap[_className])(); } };
/** * @brief 以類名為鍵的一個存放著構建方法的Hash表 * 此處一定要注意在前面添加__declspec(selectany) ,否則編譯時提示符號重定義 */ __declspec(selectany) map<string, pf> ClassMap::m_ClassMap;
/** * /class DelegatingObject RegistClassName.h? * @brief 委派模板類 */ template<typename T> class DelegatingObject? { public: /** * @brief 構造函數 * @param _className 類名 */ DelegatingObject(string _className) { ClassMap::RegistClass(_className, &(DelegatingObject::Create)); }
/** * @brief 創建實例函數 */ static CDialog* Create() { return static_cast<CDialog*>(new T); } };
/** * @brief 注冊類的宏定義 */ #ifndef REGIST_CLASS #define REGIST_CLASS(X) DelegatingObject<X> __class_##X( #X ); #endif
#endif //REGISTCLASSNAME_H
這樣只要在各自的算法對話框的Cpp文件前面添加下面一句話即可: REGIST_CLASS(CRasterTransformDlg); //柵格文件格式轉換 REGIST_CLASS(CVectorTransformDlg); //矢量文件格式轉換
然后在調用算法的地方按照下面使用方法即可,這樣就不用寫一大批的if或者switch語句了: BOOL ShowAlgoDialog(string strDlgName) { AFX_MANAGE_STATE(AfxGetStaticModuleState()); CDialog * pDlg = (ClassMap::forName(strDlgName)); if(pDlg == NULL) { AfxMessageBox("該類沒有注冊,請檢查!",MB_OK); return FALSE; } else { pDlg->DoModal(); } return TRUE; }
總結
以上是生活随笔為你收集整理的C++实现类似反射模式(模板)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 2015中国大数据技术大会在北京隆重开幕
- 下一篇: 【算法分析与设计】找到最重的球