python get()函数_C++使用ffpython嵌入和扩展python(python2和python3)

93954e242a7e79be985dd47c9e71b13a.png

C++使用ffpython嵌入和扩展python(python2和python3)

摘要:

在服务器编程中,经常会用到python脚本技术。Python是最流行的脚本之一,并且python拥有定义良好的C API接口,同时又有丰富的文档,与C++结合非常的适合。通常情况下使用C++封装机制,而用python脚本实现策略或者是控制。使用python和C++结合的技术拥有如下优势:

  • 主体系统使用C++实现,保持系统的高效。
  • 控制部分使用python,增加开发效率,python的内存垃圾回收,丰富的类库都使C++开发者获益匪浅。
  • Python脚本可以运行期重载,可以实现控制部分不停机热更新。

C++与python的编程范式有很大不同,当使用python C API调用python时,python中的一些特有机制会给C++开发者带来很多困惑。常常使用python C API时需要注意如下几点:

  • Python 使用引用计数管理内存,调用python C API时对于返回值返回的是借用的引用还是新的引用,需要根据文档仔细确认。否则轻则出现内存泄露,重则程序崩溃。
  • Python中的数据结构与C++的有很大不同。Python常用的有tuple,list,dict。而c++常用的事vector,list,map,并且c++是强类型的。当c++与python进行交互时,C++层希望操作python数据结构就像操作c++ STL一样方便,而在python脚本层,又希望c++传入的参数或返回值都是原生的python数据
  • C++中常用的指针传递对象,当嵌入python时,需要把c++对象传递到python中。 ffpython是专门方便C++嵌入python开发的类库,基于ffpython一方面可以轻松的将python集成到C++系统,另一方面,C++对象或接口也可以很容易被python使用,总之ffpython简化了c++与python的交互操作。

嵌入python

最简单的使用python的方式是把python脚本当作配置,如获取脚本中的一个字符串变量。Python的脚本文件会被python虚拟机import为module,和python的标准库的module实际上是相似的概念。ffpython封装了获取python module中的变量的操作。

printf("sys.version=%sn", ffpython.get_global_var<string>("sys", "version").c_str());

上面的代码获取python标准库中sys的version变量值,ffpython通过模板函数的自动将python的str类型自动适配到c++的string类型。get_global_var是获取变量的接口,与之对应的是设置变量的借口get_global_var:

ffpython.get_global_var("fftest", "global_var", "OhNice");
printf("fftest.global_var=%sn", ffpython.get_global_var<string>("fftest", "global_var").c_str());

调用python函数是嵌入python非常常用的操作,ffpython中提供了call接口用于调用python中的module的函数:

printf("time.asctime=%sn", ffpython.call<string>("time", "asctime").c_str());

上面的代码调用time模块的asctime方法,我们也可以使用call接口调用我们自己编写的函数:

int a1 = 100; float a2 = 3.14f; string a3 = "OhWell";
ffpython.call<void>("fftest", "test_base", a1, a2, a3);

Call被定义为模版函数,传入的参数会自动适配到python相应的类型。对应的python函数为:

def test_base(a1, a2, a3):print('test_base', a1, a2, a3)return 0

上面的python函数接受三个参数,c++传入了三个标准类型参数,实际上call接口最多支持9个泛型参数,常用的stl 参数是被支持的:

void test_stl(ffpython_t& ffpython)
{vector<int> a1;a1.push_back(100);a1.push_back(200);list<string> a2; a2.push_back("Oh");a2.push_back("Nice");vector<list<string> > a3;a3.push_back(a2);ffpython.call<bool>("fftest", "test_stl", a1, a2, a3);
}

对应调用的python函数为:

def test_stl(a1, a2, a3):print('test_stl', a1, a2, a3)return True

不但STL泛型被支持,嵌套定义的类似vector > 的结构都是被支持的,vector和list都会转换成python的list结构,而map则转换为dict结构。

调用call接口必须指定接收的返回值类型,可以使用void忽略返回值,除了可以使用标准类型,stl接口也可以被使用,python中的tuple和list可以转换成vector和list,dict则可以被转换成map。需要注意的是,若类型没有匹配,call函数将会抛出异常。用户可以catch标准异常,what接口返回的字符串包含了异常的traceback信息方便排查错误。示例如下:

try{......}catch(exception& e){printf("exception traceback %sn", e.what());}

扩展python

ffpython 可以注册static函数到python中,全局的C风格的static函数和类中定义的static函数都可以被注册到python中,示例如下:

static int print_val(int a1, float a2, const string& a3, const vector<double>& a4)
{printf("%s[%d,%f,%s,%d]n", __FUNCTION__, a1, a2, a3.c_str(), a4.size());return 0;
}
struct ops_t
{static list<int> return_stl(){list<int> ret;ret.push_back(1024);printf("%sn", __FUNCTION__);return ret;}
};std::string test_reg_function(ffpython_t& ffpython)
{ffpython.reg(&print_val, "print_val").reg(&ops_t::return_stl, "return_stl");ffpython.reg_class<foo_t, PYCTOR(int)>("foo_t").reg(&foo_t::get_value, "get_value").reg(&foo_t::set_value, "set_value").reg(&foo_t::test_stl, "test_stl").reg_property(&foo_t::m_value, "m_value");ffpython.reg_class<dumy_t, PYCTOR(int)>("dumy_t", "dumy_t class inherit foo_t ctor <int>", "foo_t").reg(&dumy_t::dump, "dump");ffpython.reg(obj_test, "obj_test");return "cppext";
}

以上代码注册了两个接口给python,然后调用fftest文件中的test_reg_function测试两个接口,fftest.py中定义测试代码:

def test_reg_function():import ext1ext1.print_val(123, 45.6 , "----789---", [3.14])ret = ext1.return_stl()print('test_reg_function', ret)

这两个接口虽然简单,但是说明了ffpython注册的接口支持多个参数,参数类型可以是标准C++类型,也可以是STL泛型。同样返回值的类型也是如此。

使用ffpython 注册C++的对象也很容易,ffpython支持注册c++类的构造函数,成员变量,成员方法到python,示例代码如下:

class foo_t
{
public:foo_t(int v_):m_value(v_){printf("%sn", __FUNCTION__);}virtual ~foo_t(){printf("%sn", __FUNCTION__);}int get_value() const { return m_value; }void set_value(int v_) { m_value = v_; }void test_stl(map<string, list<int> >& v_) {printf("%sn", __FUNCTION__);}int m_value;
};class dumy_t: public foo_t
{
public:dumy_t(int v_):foo_t(v_){printf("%sn", __FUNCTION__);}~dumy_t(){printf("%sn", __FUNCTION__);}void dump() {printf("%sn", __FUNCTION__);}
};static foo_t* obj_test(dumy_t* p)
{printf("%sn", __FUNCTION__);return p;
}

当c++类型被注册到python中后,python中使用该类型就像python内建的类型一样方便,需要注意的是,如果python中动态的创建了c++对象,那么他是被python的GC管理生命周期的,所以当变量不在被引用时,c++对象的析构函数被调用。对应的fftest.py中测试的脚本代码为:

def test_register_base_class():import ext2foo = ext2.foo_t(20130426)print("test_register_base_class get_val:", foo.get_value())foo.set_value(778899)print("test_register_base_class get_val:", foo.get_value(), foo.m_value)foo.test_stl({"key": [11,22,33] })print('test_register_base_class test_register_base_class', foo)

同前边所诉的原则相同,支持C++ 标准内建类型和STL 泛型。当这个python函数返回时,foo_t的析构函数会被调用。

dumy_t是foo_t的子类。使用ffpython可以方便表示两个类型的关系。如果基类已经定义的接口,子类不需要重复定义,比如要注册子类:

ffpython.reg_class<dumy_t, PYCTOR(int)>("dumy_t", "dumy_t class inherit foo_t ctor <int>", "foo_t").reg(&dumy_t::dump, "dump");void test_register_inherit_class(ffpython_t& ffpython)
{ffpython.call<void>("fftest", "test_register_inherit_class");
};

只需要单独注册一下子类特有的接口,其他接口自动从foo_t基类中继承而来,相应的测试python脚本代码为:

def test_register_inherit_class():import ext2dumy = ext2.dumy_t(20130426)print("test_register_inherit_class get_val:", dumy.get_value())dumy.set_value(778899)print("test_register_inherit_class get_val:", dumy.get_value(), dumy.m_value)dumy.test_stl({"key": [11,22,33] })dumy.dump()print('test_register_inherit_class', dumy)

ffpython中一个非常用用的特性是,c++创建的对象可以传递到python中,而python使用起来就像正常的python对象一样,另外python创建的c++对象也可以传递到c++中,简单示例代码:

ffpython.reg(obj_test, "obj_test");void test_cpp_obj_to_py(ffpython_t& ffpython)
{foo_t tmp_foo(2013);ffpython.call<void>("fftest", "test_cpp_obj_to_py", &tmp_foo);
}void test_cpp_obj_py_obj(ffpython_t& ffpython)
{dumy_t tmp_foo(2013);foo_t* p = ffpython.call<foo_t*>("fftest", "test_cpp_obj_py_obj", &tmp_foo);
}

相应的fftest.py中的测试脚本代码为:

def test_cpp_obj_to_py(foo):import ext2print("test_cpp_obj_to_py get_val:", foo.get_value())foo.set_value(778899)print("test_cpp_obj_to_py get_val:", foo.get_value(), foo.m_value)foo.test_stl({"key": [11,22,33] })print('test_cpp_obj_to_py test_register_base_class', foo)def test_cpp_obj_py_obj(dumy):import ext2print("test_cpp_obj_py_obj get_val:", dumy.get_value())dumy.set_value(778899)print("test_cpp_obj_py_obj get_val:", dumy.get_value(), dumy.m_value)dumy.test_stl({"key": [11,22,33] })dumy.dump()ext2.obj_test(dumy)print('test_cpp_obj_py_obj', dumy)return dumy

总结:

  • ffpython 支持c++调用python函数,获取和设置模块内的变量
  • ffpython call接口最多支持9个泛型参数,支持的类型包括c++内建的类型和STL 泛型。以及已经被注册的c++类的指针类型。返回值的类型约束同样如此。c++ STL中的vector和list对应于python的tuple和list,map类型则对应于dict。
  • ffpython支持将c++的静态函数注册到python中。
  • ffpython支持c++类的注册,并且支持继承。Python中操作c++对象就像操作原生python对象一样。
  • ffpython注册的c++类在python中被创建后,将会由python GC负责回收内存。
  • ffpython 类库只有一个文件,并且不依赖其他第三方库,非常容易集成到项目中。而且ffpython遵从开源协议。
  • ffpython使用c++模板技术,封装了python C API的使用细节,保持精巧和简洁,效率和完全的python C API编写的代码几乎相同。ffpython的实现可以作为非常好的python C API的示例。
  • Github项目地址:https://github.com/fanchy/ffpython
  • 更多文章 http://h2cloud.org

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/339928.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

鸿蒙系统的挑战,简单的讲解下何为鸿蒙系统,可能会挑战你的认知颠覆你的想象...

本帖最后由 一梦盛夏 于 2020-12-2 17:33 编辑这篇文章可能会颠覆你的想象&#xff0c;挑战你的认知&#xff0c;请耐心阅读。今天不说专业术语&#xff0c;全部用比喻方式来聊&#xff0c;这样会更明白一些&#xff0c;也欢迎大家一起来讨论。首先&#xff0c;我们先聊聊何为鸿…

卡夫卡–一次语义学

在分布式环境中&#xff0c;故障是很常见的情况&#xff0c;可以随时发生。 在Kafka环境中&#xff0c;代理可能会崩溃&#xff0c;网络故障&#xff0c;处理故障&#xff0c;发布消息时失败或无法使用消息等。这些不同的场景引入了不同类型的数据丢失和重复。 失败场景 A&am…

akka的介绍_Akka笔记–演员介绍

akka的介绍过去做过多线程的任何人都不会否认管理多线程应用程序有多么艰辛和痛苦。 我说管理是因为它开始很简单&#xff0c;一旦您开始看到性能改进&#xff0c;它就会变得非常有趣。 但是&#xff0c;当您发现没有一种简单的方法可以从子任务中的错误或难以发现的僵尸错误中…

使用模拟进行测试

如果使用正确的方法&#xff0c;模拟对象将非常有用。 我在需要驱动软件开发使用的帖子中分享了一些使用Mock Objects的经验。 在这篇文章中&#xff0c;我分享了两件事 –使用模拟进行基于合同的测试。 –用于组织模拟代码的模式。 基于合同的测试 让我们以正在构建汇款服…

aws s3 獲取所有文件_AWS SA associate 证书考试学习记录-EBS,S3,EFS比较

我们的目标&#xff0c;就是花最少的时间&#xff0c;学到最多的东西&#xff1a;-&#xff09;在AWS中&#xff0c;可以选择的存储服务很多&#xff0c;纷繁复杂&#xff0c;新手根本弄不清楚选择哪个。因为做为一个云架构师&#xff0c;你并不是让你的产品能用就够了&#xf…

html设置顶部对齐,HTML / CSS文本从div顶部对齐

首先&#xff0c;你需要修复你的CSS选择器 .你可以通过这种方式t write all those id .#content #main #services只需选择1个元素和他的孩子 .例如&#xff0c;如果你测试它&#xff0c;它将适用于你&#xff1a;#services .langelis .txt {width: 440px;height: auto;float: l…

hot编码 字符one_One Hot编码是什么?为什么要用它,什么时候用它?

作者&#xff1a;Rakshith Vasudev编译&#xff1a;ronghuaiyang导读当你在玩ML模型的时候&#xff0c;你会在任何地方遇到这个“One hot encoding”的术语。当你在玩ML模型的时候&#xff0c;你会在任何地方遇到这个“One hot encoding”术语。你可以看到一个one hot编码器的s…

CUBA 7.2 –有什么新功能?

CUBA平台的第七版向前迈出了一大步。 内部体系结构的改进和新的IDE为进一步改进奠定了良好的基础。 我们将继续添加新功能&#xff0c;以使开发人员的生活更轻松&#xff0c;并使他们的工作更加高效。 在7.2版中&#xff0c;我们引入了许多可能看起来像是主要更新的更改&#…

postgresql 分区视图_PostgreSQL架构集中式到分布式主流架构总结

文章目录一、PG未来主流架构为什么是分布式二、PostgreSQL集中式到分布式架构总结一、PG未来主流架构为什么是分布式如果说5年前DB的分布式还只是一种趋势&#xff0c;如今分布式数据库正逐渐从趋势变成主流。说到分布式&#xff0c;我想我们不能不提一下集中式和分库分表。01集…

html5 上传图片模板,HTML5实现图片文件异步上传

&#xff0c;过现前个能文使近记接的端问对字用近记接  利用HTML5的新特点做文件异步上传非常简单方便&#xff0c;本文主要展示JS部分&#xff0c;html结构。下面的代码并未使用第三发库&#xff0c;如果有参照&#xff0c;请注意一些未展现出来的代码片段。我这边的效果预览…

html中可以有两个h1,在一个HTML中h1标签能出现几次?h1标签和标题标签

首页 > web前端 > html教程 > 正文 在一个HTML中h1标签能出现几次&#xff1f;h1标签和标题标签的差别是什么&#xff1f; 2018-08-29 10:57:28本篇文章主要介绍了关于HTML h1标签的一些解释&#xff0c;有html h1标签和html title标签的区别&#xff0c;还有网页中h1…

Java中的记录类型

2020年3月发布的JDK 14引入了记录 &#xff08;预览语言功能&#xff09;&#xff0c;这些记录提供了一种紧凑的语法来声明主要用于保存数据的类。 在记录中 &#xff0c;所有低级&#xff0c;重复且容易出错的代码都类似于构造函数&#xff0c;访问器和通用方法&#xff0c;例…

山东省102021年普通高考成绩查询,山东高考成绩今日发布!成绩查询看这里!

原标题&#xff1a;山东高考成绩今日发布&#xff01;成绩查询看这里&#xff01;山东高考生注意啦~今天16:20举行山东2020年夏季高考第二次新闻发布会届时将会公布高考录取政策、分数线情况等今天17:00公布2020夏季高考与等级考成绩发布会怎么看&#xff1f;高考成绩怎样查&am…

使用SoapUI调用不同的安全WCF SOAP服务-基本身份验证,第二部分

在本系列的第一篇文章中&#xff0c;我们创建了一个基本的身份验证服务&#xff0c;以使用SoapUI进行调用。 因此&#xff0c;在第二篇文章中&#xff0c;我们将逐步演示如何使用此工具成功调用这种服务。 使用SoapUI的1-Basic WCF SOAP –创建新的SOAP项目 首先&#xff0c;我…

html table nei边框线,GitHub - meichuanneiku/TableCell: 在TableBank的基础上,进一步标注到单元格精度,利用目标检测/分割实现单元格定位。...

项目说明本项目是我2019年7月份的实习工作的**展示与记录**&#xff1a;把倾斜的表格旋转水平&#xff1b;制作5000张表格数据集&#xff0c;需要标注每一个单元格&#xff0c;并实现单元格检测第一项比较简单&#xff0c;仿射变换、透视变换已经很成熟了&#xff0c;关键是第二…

前缀命名

如果您是第一次查看Takes或Cactoos的源代码&#xff0c;则很可能会像其他名称一样被命名约定触发&#xff0c;这意味着大多数类名称都有两个字母的前缀&#xff1a; BkSafe &#xff0c; RqFake &#xff0c; RsWithStatus &#xff0c; TkGzip等。 老实说&#xff0c;我还没有…

再访PMML

嗨伙计&#xff01; 从今年年初开始&#xff0c;就有了重新设计Drools PMML模块的计划。 在这篇文章中&#xff0c;我将描述我们将如何处理它&#xff0c;目前的状态&#xff0c;未来发展的想法等&#xff0c;等等……敬请期待&#xff01; 背景 PMML是一个标准&#xff0c;旨…

用计算机怎么弹离人愁数字,拇指琴新手入门曲谱——离人愁

喜欢古风的朋友赶快凑过来啦&#xff0c;最近抖音上超火的离人愁拇指琴教学&#xff0c;喜欢离人愁的小姐姐小哥哥赶快学起来啦&#xff01;以下琴谱适用于Hugh Tracey G调17键。南非琴出厂调音是G调排列&#xff0c;习惯了C调音阶排列的朋友可能对G调排列不是很适应。因为两者…

jvm7 jvm8_JVM PermGen –您在哪里?

jvm7 jvm8这篇文章介绍了JVM内存结构的一些基础知识&#xff0c;并快速窥视了PermGen&#xff0c;以了解自Java SE 8出现以来它已消失的地方。 裸基础 JVM只是系统上运行的另一个进程&#xff0c;魔术始于java命令。 像任何OS进程一样&#xff0c;它需要内存才能运行。 请记住…

C++类与对象(中)第二篇

前言&#xff1a; C语言中使用运算符是对内置类型的数据进行操作&#xff0c;但是在C中有了对象&#xff0c;导致对象无法通过运算符进行运算&#xff0c;故引入了运算符重载即需要重新定义这些运算符&#xff0c;赋予已有运算符新的功能&#xff0c;使它能够用于特定类型执行特…