免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 4987 | 回复: 12
打印 上一主题 下一主题

[C++] C++ 如何反射调用方法 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-12-10 10:58 |只看该作者 |倒序浏览
Hi~最近想实现一个简单的C++反射应用,根据URI调用指定对象的指定方法,大致如下:
  1. //@(path="/api/ubuntu")
  2. DECLARE_DYN_CLASS(Test) {
  3. public:
  4.   Test()
  5.   { }

  6.   //@(path="/display", method="UPDATE")
  7.   void display()
  8.   { std::cout << "Test::display()" << std::endl; }
  9. };

  10. //@(path="/api/lxc")
  11. DECLARE_DYN_CLASS(Api) {
  12. public:
  13.   Api()
  14.   { }

  15.   //@(path="/create", method="POST")
  16.   void do_post()
  17.   { std::cout << "Api::do_post()" << std::endl; }

  18.   //@(path="/create", method="GET")
  19.   void do_get()
  20.   { std::cout << "Api::do_get()" << std::endl; }  
  21. };

  22. int main()
  23. {
  24.   Test* test = (Test*)ClassFactory::create_instance_for_name("Test");
  25.   test->display();

  26.   Api* api = (Api*)ClassFactory::create_instance_for_name("Api");
  27.   api->do_post();
  28.   api->do_get();

  29.   return 0;
  30. }
复制代码
问题是不知道如何反射调用类的成员函数,目前想到的方法是:
typedef void (ClassType::*PMF)(std::string request);
然后将成员函数指针强制转换为int*存到一个map里,可是会出现
链接错误,折腾了好几天了,哪位高手给指点迷津下~多谢了^^
Demo.rar (55.68 KB, 下载次数: 4)

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
2 [报告]
发表于 2013-12-11 10:09 |只看该作者
Test* test = (Test*)ClassFactory::create_instance_for_name("Test");
test->display();
----------
这样的语法,表明Test与是否反射没有关系。反射应该是
Foo * f = create_instance("test");
f->invoke("display");

论坛徽章:
0
3 [报告]
发表于 2013-12-11 10:21 |只看该作者
本帖最后由 ljpdxj 于 2013-12-11 10:22 编辑

楼上太高深了!

论坛徽章:
0
4 [报告]
发表于 2013-12-11 10:47 |只看该作者
其实COM也算是反射吧,所有的函数都在IDL里写着。换成spring,只不过改成 xml了。不过spring可以赋值,这东西和程序读配置文件有多大区别呢

论坛徽章:
0
5 [报告]
发表于 2013-12-11 11:22 |只看该作者
回复 2# truekbcl

确实如此,
目前我能做到的:
1)类对应的URI 映射到 类名;方法对应的URI和METHOD 映射到 方法名
2)类名 映射到 创建实例的方法指针;方法名 映射到 成员函数指针
接下来如何组装不知道了。。。其实不一定要达到真正的反射,只是想用C++简单模拟Spring根据注解装配对象和引导请求的过程,
还希望能给点进一步的建议,感激不尽


   

论坛徽章:
24
狮子座
日期:2013-12-31 10:48:0015-16赛季CBA联赛之吉林
日期:2016-04-18 14:43:1015-16赛季CBA联赛之北控
日期:2016-05-18 15:01:4415-16赛季CBA联赛之上海
日期:2016-06-22 18:00:1315-16赛季CBA联赛之八一
日期:2016-06-25 11:02:2215-16赛季CBA联赛之佛山
日期:2016-08-17 22:48:2615-16赛季CBA联赛之福建
日期:2016-12-27 22:39:272016科比退役纪念章
日期:2017-02-08 23:49:4315-16赛季CBA联赛之八一
日期:2017-02-16 01:05:3415-16赛季CBA联赛之山东
日期:2017-02-22 15:34:5615-16赛季CBA联赛之上海
日期:2017-11-25 16:17:5015-16赛季CBA联赛之四川
日期:2016-01-17 18:38:37
6 [报告]
发表于 2013-12-11 12:09 |只看该作者
回复 1# buaaspy


    这事还是改编译器比较好。

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
7 [报告]
发表于 2013-12-11 12:33 |只看该作者
高端!大气!上档次!

论坛徽章:
0
8 [报告]
发表于 2013-12-11 12:44 |只看该作者
这个问题的难点在于可变参数的封装,因为成员函数的参数是不确定的。
使用map的方式,只能针对特定签名形式的成员函数进行封装,而无法提供统一的接口。

可以考虑使用代码生成技术,使用python之类的脚本语言读入定义的C++文件,将class解析出来,然后针对各个成员函数生成Invoke方法。

buaaspy 发表于 2013-12-11 11:22
回复 2# truekbcl

确实如此,

论坛徽章:
44
15-16赛季CBA联赛之浙江
日期:2021-10-11 02:03:59程序设计版块每日发帖之星
日期:2016-07-02 06:20:0015-16赛季CBA联赛之新疆
日期:2016-04-25 10:55:452016科比退役纪念章
日期:2016-04-23 00:51:2315-16赛季CBA联赛之山东
日期:2016-04-17 12:00:2815-16赛季CBA联赛之福建
日期:2016-04-12 15:21:2915-16赛季CBA联赛之辽宁
日期:2016-03-24 21:38:2715-16赛季CBA联赛之福建
日期:2016-03-18 12:13:4015-16赛季CBA联赛之佛山
日期:2016-02-05 00:55:2015-16赛季CBA联赛之佛山
日期:2016-02-04 21:11:3615-16赛季CBA联赛之天津
日期:2016-11-02 00:33:1215-16赛季CBA联赛之浙江
日期:2017-01-13 01:31:49
9 [报告]
发表于 2013-12-11 13:52 |只看该作者
本帖最后由 windoze 于 2013-12-11 13:57 编辑

其实吧,你为什么一定要把这些东西塞到一个class里呢?
为什么不干脆写成这样:

  1. typedef std::map<std::pair<std::string, std::string>, std::function<void()>> action_map;
  2. action_map the_map({
  3.         {
  4.                 {"GET", "/some/path"},
  5.                 [](){ std::cout << "GET /some/path" << std::endl; }
  6.         },
  7.         {
  8.                 {"POST", "/some/path"},
  9.                 [](){ std::cout << "POST /some/path" << std::endl; }
  10.         },
  11.         {
  12.                 {"GET", "/some/other/path"},
  13.                 [](){ std::cout << "GET /some/other/path" << std::endl; }
  14.         },
  15. });
复制代码
使用的时候你可以这样:

  1. action_map::const_iterator i=the_map.find({method, path});
  2. if(i==the_map.end()) {
  3.     // Send 404 response
  4. } else {
  5.     i->second();
  6. }
复制代码

论坛徽章:
0
10 [报告]
发表于 2013-12-11 18:09 |只看该作者
回复 8# yao050421103

好主意!
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP