从实例重温工厂模式和单件模式 收藏
本文由恋花蝶最初发表于:http://blog.csdn.net/lanphaday,你可以在保持文章完整和保留本声明的情况下转帖、分发和印刷等。
http://community.csdn.net/Expert/topic/4892/4892570.xml?temp=.8727381
今天一个好朋友发了上面这个贴,并邀我过去看看。就去看了看,当时觉得应该用工厂模式去解决,随便写了几句上去,后来又有冲动想写一下代码,正好重温一下经典的工厂模式,就写了如下代码。主要应用了工厂模式和单件模式。
现在如果要增加新的指令需要修改的地方也是相当地少:只要继承自CCommand,并实现自己的处理函数,然后向类工厂登记一下就可以了。
废话不多说了,看代码吧。感觉写这个东西学到不少东西,所谓温故而知新嘛。
先看问题:
有一个脚本文件,每行放一个命令,格式如下:
Name[,Parameter[,Value]]
其中,Parameter 和 Value 是可以省略的。Command分为不同的种类,对于特定的种类,Parameter 和 Value 是不是出现是一定的,如 "START"命令就没有Parameter 和 Value ,而"SEND"命令就有。头和我说,用类的层次结构来表示,外面在套一个解析类。我是这样做的
class CCommand{
string Name;
string Parameter;
string Value;
};
那么,按照我们头的说法,我需要:
class CStartCommang : public CCommand{
//在这里我需要做什么呢??
}
解析类CScriptExplain又该怎么定义呢?
我的参考代码,拷贝到一个.cpp文件中,即可编译、执行。我用的是VC6,相信MS的编译器也可以,GCC、BCB、DevC++应该也行。
#pragma warning(disable:4786) #include <iostream> #include <cassert> #include <string> #include <map> #include <fstream> //也可以采用构造函数无参,doCommand有参的方式,那样就可以全部CCommand的派生类做成单件。 class CCommand { protected: CCommand(){} CCommand(const std::string& strCmd):m_strCmd(strCmd){} public: virtual ~CCommand(){} virtual void doCommand()=0; static CCommand* CreateInstance(const std::string& strCmd); const std::string& GetCommandString(){return m_strCmd;} private: std::string m_strCmd; }; class CStartCommand:public CCommand { private: CStartCommand(){} CStartCommand(const std::string& strCmd):CCommand(strCmd) { //在这里解释strCmd里的Param和Value等 } public: virtual ~CStartCommand(){} void doCommand() { //在这里做这个命令要做的事 std::cout << "Doing Start Command!/t" << GetCommandString() << std::endl; } static CCommand* CreateInstance(const std::string& strCmd) { CCommand* pCmd = new CStartCommand(strCmd); return pCmd; } }; class CSendCommand:public CCommand { private: CSendCommand(){} CSendCommand(const std::string& strCmd):CCommand(strCmd) { //在这里解释strCmd里的Param和Value等 } public: virtual ~CSendCommand(){} void doCommand() { //在这里做这个命令要做的事 std::cout << "Doing Send Command!/t" << GetCommandString() << std::endl; } static CCommand* CreateInstance(const std::string& strCmd) { CCommand* pCmd = new CSendCommand(strCmd); return pCmd; } }; typedef CCommand*(*CreateInstanceFunc)(const std::string&); class CCommandFactory { private: CCommandFactory(){} public: virtual ~CCommandFactory(){} static CCommandFactory* GetInstance() { static CCommandFactory ins = CCommandFactory(); return &ins; } void AddCommandFunc(std::string& strCmdName, CreateInstanceFunc func) { m_CommandFuncMap[strCmdName] = func; } CCommand* CreateCommand(std::string& strCmdName, std::string strCmd) { return m_CommandFuncMap[strCmdName](strCmd); } void ReleaseCommand(CCommand* p) { assert(p); delete p; } private: std::map<std::string, CreateInstanceFunc> m_CommandFuncMap; }; class CScriptExplain { public: void doFile(const char* strFileName) { CCommandFactory* pFactory = CCommandFactory::GetInstance(); std::ifstream f(strFileName); #define MAX_CMD_LEN 256 char strCmd[MAX_CMD_LEN]; char tmp[MAX_CMD_LEN]; do{ f.getline(strCmd, MAX_CMD_LEN); if(strlen(strCmd)==0)break; int idx = 0; for(; idx < MAX_CMD_LEN; ++idx) { if(strCmd[idx]==','||strCmd[idx]=='/0')break; tmp[idx] = strCmd[idx]; } if(idx == MAX_CMD_LEN)break; tmp[idx] = '/0'; CCommand* pCmd = pFactory->CreateCommand(std::string(tmp), std::string(strCmd+idx)); pCmd->doCommand(); pFactory->ReleaseCommand(pCmd); }while(true); } }; int main() { CCommandFactory* pFactory = CCommandFactory::GetInstance(); pFactory->AddCommandFunc(std::string("Start"), CStartCommand::CreateInstance); pFactory->AddCommandFunc(std::string("Send"), CSendCommand::CreateInstance); CScriptExplain se; se.doFile("my_script.txt"); return 0; }
my_script.txt的内容:
Start
Send,Port,5555
Send,Port,8888
Send,Port,8673