c++的反射的demo

c++的反射的demo

一个基本例子

reflection.cpp

// 工厂类的定义
#include <map>
#include <iostream>
#include <string>
using namespace std;
typedef void* (*PTRCreateObject)(void);
class ClassFactory {
private:
    map<string, PTRCreateObject> m_classMap ;
    ClassFactory(){}; //构造函数私有化

public:
    void* getClassByName(string className);
    void registClass(string name, PTRCreateObject method) ;
    static ClassFactory& getInstance() ;
};
// 工厂类的实现
// brief: 获取工厂类的单个实例对象
ClassFactory& ClassFactory::getInstance(){
    static ClassFactory sLo_factory;
    return sLo_factory ;
}

// brief: 通过类名称字符串获取类的实例
void* ClassFactory::getClassByName(string className){
    map<string, PTRCreateObject>::const_iterator iter;
    iter = m_classMap.find(className) ;
    if ( iter == m_classMap.end() )
        return NULL ;
    else
        return iter->second() ;
}
// brief: 将给定的类名称字符串和对应的创建类对象的函数保存到map中
void ClassFactory::registClass(string name, PTRCreateObject method){
    m_classMap.insert(pair<string, PTRCreateObject>(name, method)) ;
}
class RegisterAction {
public:
    RegisterAction(string className,PTRCreateObject ptrCreateFn) {
        ClassFactory::getInstance().registClass(className, ptrCreateFn);
    }
};
#define REGISTER(className)                                             \
    className* objectCreator##className(){                              \
        return new className;                                           \
    }                                                                   \
    RegisterAction g_creatorRegister##className(                        \
        #className,(PTRCreateObject)objectCreator##className)


// test class
class TestClass {
public:
    void m_print() {
        cout<<"hello TestClass"<<endl;
    };
};
REGISTER(TestClass);

int main(int argc,char* argv[]) {
    TestClass* ptrObj=(TestClass*)ClassFactory::getInstance().getClassByName("TestClass");
    ptrObj->m_print();
}

把这个改造成:调用方不改,底层so可以改的方式

先定义反射工具类
reflection.h

// 工厂类的定义
#ifndef __hao_ClassFactory__
#define __hao_ClassFactory__
#include <map>
#include <string>
using namespace std;
typedef void* (*PTRCreateObject)(void);
class ClassFactory {
private:
    map<string, PTRCreateObject> m_classMap ;
    ClassFactory(){}; //构造函数私有化

public:
    void* getClassByName(string className);
    void registClass(string name, PTRCreateObject method) ;
    static ClassFactory& getInstance() ;
};

class RegisterAction {
public:
    RegisterAction(string className,PTRCreateObject ptrCreateFn) {
        ClassFactory::getInstance().registClass(className, ptrCreateFn);
    }
};
#define REGISTER(className)                                             \
    className* objectCreator##className(){                              \
        return new className;                                           \
    }                                                                   \
    RegisterAction g_creatorRegister##className(                        \
        #className,(PTRCreateObject)objectCreator##className)

#endif

实现类:

// 工厂类的定义
#include <map>
#include <iostream>
#include <string>
#include <reflection.h>
// 工厂类的实现
// brief: 获取工厂类的单个实例对象
ClassFactory& ClassFactory::getInstance(){
    static ClassFactory sLo_factory;
    return sLo_factory ;
}
// brief: 通过类名称字符串获取类的实例
void* ClassFactory::getClassByName(string className){
    map<string, PTRCreateObject>::const_iterator iter;
    iter = m_classMap.find(className) ;
    if ( iter == m_classMap.end() )
        return NULL ;
    else
        return iter->second() ;
}
// brief: 将给定的类名称字符串和对应的创建类对象的函数保存到map中
void ClassFactory::registClass(string name, PTRCreateObject method){
    m_classMap.insert(pair<string, PTRCreateObject>(name, method)) ;
}

生成工具类动态库

g++ -std=c++11 -fpic -shared -o libreflection.so reflection.cpp -I.

定义接口

#ifndef __TestClass__
#define __TestClass__

#include <map>
#include <iostream>
#include <string>
using namespace std;

// test class
class TestClass {
public:
    virtual void m_print() = 0;
//    void m_print() {
//        cout<<"hello TestClass"<<endl;
//    };
};
//REGISTER(TestClass);

#endif

主函数为

#include <map>
#include <iostream>
#include <string>
#include <reflection.h>
#include <TestClass.h>
#include <main.h>
using namespace std;

int main(int argc,char* argv[]) {
   cout << "argv 1: "<<  argv[1] << endl;
   //TestClass* ptrObj=(TestClass*)ClassFactory::getInstance().getClassByName("TestClassA");
   TestClass* ptrObj=(TestClass*)ClassFactory::getInstance().getClassByName(argv[1]);
   ptrObj->m_print();
}

main.h里只定义:

#include <TestClassA.h>
#include <TestClassB.h>

具体的实现TestClassA.h为

#ifndef __TestClassA__
#define __TestClassA__
#include <TestClass.h>
#include <reflection.h>
#include <iostream>
using namespace std;
class TestClassA : public TestClass
{
public:
    void m_print();
};
REGISTER(TestClassA);
#endif

TestClassA.cpp

#include <TestClassA.h>
#include <iostream>
using namespace std;
void TestClassA::m_print(){
        cout<<"hello TestClassA...."<<endl;
}

TestClassB.h

#ifndef __TestClassB__
#define __TestClassB__
#include <TestClass.h>
#include <reflection.h>
#include <iostream>
//TestClassB {
using namespace std;
class TestClassB : public TestClass{

public:
    void m_print();
};
REGISTER(TestClassB);
#endif

TestClassB.cpp

#include <TestClassB.h>
#include <iostream>
using namespace std;
void TestClassB::m_print(){
        cout<<"hello TestClassB....i am changed...."<<endl;
}

编译TestClassA:

#in linux
g++ -std=c++11 -fpic -shared -o libclassa.so TestClassA.cpp -I.
#in mac:
g++ -std=c++11 -fpic -shared -o libclassa.so TestClassA.cpp -I. -L. -lreflection

编译TestClassB

#in linux
g++ -std=c++11 -fpic -shared -o libclassb.so TestClassB.cpp -I.
#in mac
g++ -std=c++11 -fpic -shared -o libclassb.so TestClassB.cpp -I. -L. -lreflection

编译主程序:

g++ -g -std=c++11   main.cpp -I. -L. -lclassa -lclassb -lreflection -o main

注意:如果so库找不到
export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH

测试

#!/bin/sh
./main TestClassA
./main TestClassB

结果:

[root@4c8g210129 demo2]# ./test.sh
argv 1: TestClassA
hello TestClassA....
argv 1: TestClassB
hello TestClassB....i am changed....
[root@4c8g210129 demo2]#

每次修改TestClassA.cpp或者TestClassB.cpp
的时候不用修改main.cpp

上一篇:Jenkins + Svn + Ant持续集成(增量包处理)


下一篇:如何用mocha找到子组件的className