先贴一段代码,再慢慢解释

  1. #include <python2.7/Python.h>
  2. #include <stdio.h>
  3. #include <string.h>
  4. void printDict(PyObject* obj) {
  5. int i = 0;
  6. if (!PyDict_Check(obj))
  7. return;
  8. PyObject *k, *keys;
  9. keys = PyDict_Keys(obj);
  10. for (i = 0; i < PyList_GET_SIZE(keys); i++) {
  11. k = PyList_GET_ITEM(keys, i);
  12. char* c_name = PyString_AsString(k);
  13. printf("%s\n", c_name);
  14. }
  15. }
  16. int main() {
  17. //启动python解析器,初始化环境
  18. Py_Initialize();
  19. if (!Py_IsInitialized())
  20. return -1;
  21. PyRun_SimpleString("import sys");
  22. PyRun_SimpleString("sys.path.append('./login')");
  23. //导入模块
  24. PyObject* pModule = PyImport_ImportModule("ssh");
  25. if (!pModule) {
  26. printf("Cannot open python file!\n");
  27. return -1;
  28. }
  29. //模块的字典列表
  30. PyObject* pDict = PyModule_GetDict(pModule);
  31. if (!pDict) {
  32. printf("Cannot find dictionary.\n");
  33. return -1;
  34. }
  35. //打印出来看一下
  36. printDict(pDict);
  37. //演示函数调用
  38. //    PyObject* pFunHi = PyDict_GetItemString(pDict, "sayhi");
  39. //
  40. //    PyObject_CallFunction(pFunHi, "s", "lhb");
  41. //
  42. //    Py_DECREF(pFunHi);
  43. //演示构造一个Python对象,并调用Class的方法    PyObject* pClassLogin = PyDict_GetItemString(pDict, "login");
  44. if (!pClassLogin) {
  45. printf("Cannot find login class.\n");
  46. return -1;
  47. }
  48. PyObject * pArgs = PyTuple_New(3);
  49. //    PyTuple_SetItem(pArgs, 0, Py_BuildValue("s", "ssh"));
  50. PyTuple_SetItem(pArgs, 0, Py_BuildValue("s", "10.20.60.23"));
  51. PyTuple_SetItem(pArgs, 1, Py_BuildValue("s", "root"));
  52. PyTuple_SetItem(pArgs, 2, Py_BuildValue("s", "nsfocus"));
  53. //    PyTuple_SetItem(pArgs, 4, Py_BuildValue("s", "22"));
  54. PyObject* pInstanceLogin = PyInstance_New(pClassLogin, pArgs, NULL);
  55. if (!pInstanceLogin) {
  56. printf("Cannot find login instance.\n");
  57. return -1;
  58. }
  59. PyObject * ret = PyObject_CallMethod(pInstanceLogin, "execute_cmd", "(s)", "uname -a");
  60. char * str_ret = PyString_AsString(ret);
  61. printf("result:%s\n", str_ret);
  62. //char str_ret1[1000];
  63. //PyArg_Parse((ret),"s",&str_ret1);
  64. //printf("result1:%s\n", str_ret1);
  65. //释放
  66. Py_DECREF(pInstanceLogin);
  67. Py_DECREF(pClassLogin);
  68. Py_DECREF(pModule);
  69. Py_Finalize();
  70. return 0;

命令行下运行:gcc c_py.c -L/usr/lib/ -lpython2.7 -o c_py,

运行结果如下:

  1. OptiPlex-790:~/example_code$ ./c_py
  2. libssh2
  3. socket
  4. __builtins__
  5. __file__
  6. __package__
  7. sys
  8. re
  9. __name__
  10. login
  11. __doc__
  12. 10.20.60.23
  13. init socket..
  14. init session..
  15. reading...
  16. result:Linux sunorr 2.6.24-16-generic #1 SMP Thu Apr 10 13:23:42 UTC 2008 i686 GNU/Linux
c调用python
Py_Initialize()和 Py_Finalize()是初始和销毁Python解释器
PyRun_SimpleString("import sys") 导入sys,这是c中执行python语句的一个接口
PyImport_ImportModule("ssh") 导入ssh.py这个模块
PyModule_GetDict(pModule)获取模块的字典列表,通过这个字典就能获取整个模块的结构,和直接dir(ssh)的效果事一样的。这个时候就可以通过这个dict去获取模块中的函数,变量,类。
PyDict_GetItemString(pDict, "sayhi")获取模块中的函数
PyObject_CallFunction(pFunHi, "s", "lhb")运行这个函数
PyObject * pArgs = PyTuple_New(3);//生成一个tuple
PyTuple_SetItem(pArgs, 0, Py_BuildValue("s", "10.20.60.23"),Py_BuildValue就是将c中的结构转变为python中的结构,设置第一个参数为某ip。
PyObject* pInstanceLogin = PyInstance_New(pClassLogin, pArgs, NULL),获取一个运行实例,第一个参数是类的地址,第二个参数是参数,第三个参数是key-alue类型参数
PyObject * ret = PyObject_CallMethod(pInstanceLogin, "execute_cmd", "(s)", "uname -a")
通过实例中execute_cmd方法去执行命令

char * str_ret = PyString_AsString(ret);
printf("result:%s\n", str_ret)

获取返回值

另外返回值的转换也可采用PyArg_ParseTuple

如果需要在C程序中用Python脚本传递参数,或者获得Python脚本的返回值,则要使用更多的函数来编写C程序。由于Python有自己的数据类型,因此在C程序中要使用专门的API对相应的数据类型进行操作。常用的函数有以下几种。

1.数字与字符串处理

在Python/C API中提供了Py_BuildValue()函数对数字和字符串进行转换处理,使之变成Python中相应的数据类型。其函数原型如下所示。

PyObject* Py_BuildValue( const char *format, ...)

其参数含义如下。

·     format:格式化字符串,如表8-1所示。

Py_BuildValue()函数中剩余的参数即要转换的C语言中的整型、浮点型或者字符串等。其返回值为PyObject型的指针。在C语言中,所有的Python类型都被声明为PyObject型。

2.列表操作

在Python/C API中提供了PyList_New()函数用以创建一个新的Python列表。PyList_New()函数的返回值为所创建的列表。其函数原型如下所示。

PyObject* PyList_New( Py_ssize_t len)

其参数含义如下。

·     len:所创建列表的长度。

当列表创建以后,可以使用PyList_SetItem()函数向列表中添加项。其函数原型如下所示。

int PyList_SetItem( PyObject *list, Py_ssize_t index, PyObject *item)

其参数含义如下。

·     list:要添加项的列表。

·     index:所添加项的位置索引。

·     item:所添加项的值。

同样可以使用Python/C API中PyList_GetItem()函数来获取列表中某项的值。PyList_GetItem()函数返回项的值。其函数原型如下所示。

PyObject* PyList_GetItem( PyObject *list, Py_ssize_t index)

其参数含义如下。

·     list:要进行操作的列表。

·     index:项的位置索引。

Python/C API中提供了与Python中列表操作相对应的函数。例如列表的append方法对应于PyList_Append()函数。列表的sort方法对应于PyList_Sort()函数。列表的reverse方法对应于PyList_Reverse()函数。其函数原型分别如下所示。

int PyList_Append( PyObject *list, PyObject *item)

int PyList_Sort( PyObject *list)

int PyList_Reverse( PyObject *list)

对于PyList_Append()函数,其参数含义如下。

·     list:要进行操作的列表。

·     item:要参加的项。

对于PyList_Sort()和PyList_Reverse()函数,其参数含义相同。

·     list:要进行操作的列表。

3.元组操作

在Python/C API中提供了PyTuple_New()函数,用以创建一个新的Python元组。PyTuple_New()函数返回所创建的元组。其函数原型如下所示。

PyObject* PyTuple_New( Py_ssize_t len)

其参数含义如下。

·     len:所创建元组的长度。

当元组创建以后,可以使用PyTuple_SetItem()函数向元组中添加项。其函数原型如下所示。

int PyTuple_SetItem( PyObject *p, Py_ssize_t pos, PyObject *o)

其参数含义如下所示。

·     p:所进行操作的元组。

·     pos:所添加项的位置索引。

·     o:所添加的项值。

可以使用Python/C API中PyTuple_GetItem()函数来获取元组中某项的值。PyTuple_GetItem()函数返回项的值。其函数原型如下所示。

PyObject* PyTuple_GetItem( PyObject *p, Py_ssize_t pos)

其参数含义如下。

·     p:要进行操作的元组。

·     pos:项的位置索引。

当元组创建以后可以使用_PyTuple_Resize()函数重新调整元组的大小。其函数原型如下所示。

int _PyTuple_Resize( PyObject **p, Py_ssize_t newsize)

其参数含义如下。

·     p:指向要进行操作的元组的指针。

·     newsize:新元组的大小。

4.字典操作

在Python/C API中提供了PyDict_New()函数用以创建一个新的字典。PyDict_New()函数返回所创建的字典。其函数原型如下所示。

PyObject* PyDict_New()

当字典创建后,可以使用PyDict_SetItem()函数和PyDict_SetItemString()函数向字典中添加项。其函数原型分别如下所示。

int PyDict_SetItem( PyObject *p, PyObject *key, PyObject *val)

int PyDict_SetItemString( PyObject *p, const char *key, PyObject *val)

其参数含义如下。

·     p:要进行操作的字典。

·     key:添加项的关键字,对于PyDict_SetItem()函数其为PyObject型,对于PyDict_SetItemString()函数其为char型。

·     val:添加项的值。

使用Python/C API中的PyDict_GetItem()函数和PyDict_GetItemString()函数来获取字典中某项的值。它们都返回项的值。其函数原型分别如下所示。

PyObject* PyDict_GetItem( PyObject *p, PyObject *key)

PyObject* PyDict_GetItemString( PyObject *p, const char *key)

其参数含义如下。

·     p:要进行操作的字典。

·     key:添加项的关键字,对于PyDict_GetItem()函数其为PyObject型,对于PyDict_GetItemString()函数其为char型。

使用Python/C API中的PyDict_DelItem()函数和PyDict_DelItemString()函数可以删除字典中的某一项。其函数原型如下所示。

int PyDict_DelItem( PyObject *p, PyObject *key)

int PyDict_DelItemString( PyObject *p, char *key)

其参数含义如下。

·     p:要进行操作的字典。

·     key:添加项的关键字,对于PyDict_DelItem()函数其为PyObject型,对于PyDict_DelItemString()函数其为char型。

使用Python/C API中的PyDict_Next()函数可以对字典进行遍历。其函数原型如下所示。

int PyDict_Next( PyObject *p, Py_ssize_t *ppos, PyObject **pkey, PyObject **pvalue)

其参数含义如下。

·     p:要进行遍历的字典。

·     ppos:字典中项的位置,应该被初始化为0。

·     pkey:返回字典的关键字。

·     pvalue:返回字典的值。

在Python/C API中提供了与Python中字典操作相对应的函数。例如字典的item方法对应于PyDict_Items()函数。字典的keys方法对应于PyDict_Keys()函数。字典的values方法对应于PyDict_Values()函数。其函数原型分别如下所示。

PyObject* PyDict_Items( PyObject *p)

PyObject* PyDict_Keys( PyObject *p)

PyObject* PyDict_Values( PyObject *p)

其参数含义如下。

·     p:要进行操作的字典。

5.释放资源

Python使用引用计数机制对内存进行管理,实现自动垃圾回收。在C/C++中使用Python对象时,应正确地处理引用计数,否则容易导致内存泄漏。在Python/C API中提供了Py_CLEAR()、Py_DECREF()等宏来对引用计数进行操作。

当使用Python/C API中的函数创建列表、元组、字典等后,就在内存中生成了这些对象的引用计数。在对其完成操作后应该使用Py_CLEAR()、Py_DECREF()等宏来销毁这些对象。其原型分别如下所示。

void Py_CLEAR( PyObject *o)

void Py_DECREF( PyObject *o)

其参数含义如下。

·     o:要进行操作的对象。

对于Py_CLEAR()其参数可以为NULL指针,此时,Py_CLEAR()不进行任何操作。而对于Py_DECREF()其参数不能为NULL指针,否则将导致错误。

6.模块与函数

使用Python/C API中的PyImport_Import()函数可以在C程序中导入Python模块。PyImport_Import()函数返回一个模块对象。其函数原型如下所示。

PyObject* PyImport_Import( PyObject *name)

其参数含义如下。

·     name:要导入的模块名。

使用Python/C API中的PyObject_CallObject()函数和PyObject_CallFunction()函数,可以在C程序中调用Python中的函数。其参数原型分别如下所示。

PyObject* PyObject_CallObject( PyObject *callable_object, PyObject *args)

PyObject* PyObject_CallFunction( PyObject *callable, char *format, ...)

对于PyObject_CallObject()函数,其参数含义如下。

·     callable_object:要调用的函数对象。

·     args:元组形式的参数列表。

对于PyObject_CallFunction()函数,其参数含义如下。

·     callable_object:要调用的函数对象。

·     format:指定参数的类型。

·     ...:向函数传递的参数。

使用Python/C API中的PyModule_GetDict()函数可以获得Python模块中的函数列表。PyModule_GetDict()函数返回一个字典。字典中的关键字为函数名,值为函数的调用地址。其函数原型如下所示。

PyObject* PyModule_GetDict( PyObject *module)

其参数含义如下。

·     module:已导入的模块对象。

转载于:https://blog.51cto.com/3502990/1117983

c嵌入python类获取返回值相关推荐

  1. c++ 调用python2类获取返回值

    vs2017好像没有 PyObject* PyIns = PyInstance_New(pyCls, NULL, NULL); // 创建tester类 前面介绍了用python调用C++时用swig ...

  2. python类继承返回值_python继承threading.Thread实现有返回值的子类实例

    继承与threading.thread实现有返回值的子类mythread,废话不多说,大家直接看代码 import threading class mythread(threading.thread) ...

  3. python curl 获取返回值_python获取系统命令的返回值 | curl in python

    项目背景: 通过curl 命令 从服务器获取数组,tids,然后导入到Mysql 表中. 自己想的方案: shell命令不熟悉,另外python中的数据库等接口都已封装,所以想使用Python来做,利 ...

  4. python curl 获取返回值_python-将curl查询转换为请求

    我试图在以下位置使用文档: https://pairbulkdata.uspto.gov/#/api-documentation 但是,当我尝试这些查询时,出现错误消息. 我正在尝试将curl查询转换 ...

  5. 获取python脚本的返回值_Python多线程获取返回值代码实例

    这篇文章主要介绍了Python多线程获取返回值代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 在使用多线程的时候难免想要获取其操作完的返回值 ...

  6. python os.system重定向stdout到变量 ,同时获取返回值

    python os.system重定向stdout到变量 ,同时获取返回值 Python执行系统命令的方法 os.system(),os.popen(),commands 最近在做那个测试框架的时候发 ...

  7. python 调用控制台并获取返回结果_Java调用Python脚本并获取返回值

    在Java程序中有时需要调用Python的程序,这时可以使用一般的PyFunction来调用python的函数并获得返回值,但是采用这种方法有可能出现一些莫名其妙的错误,比如ImportError.在 ...

  8. 使用java运行Python脚本并获取返回值

    java运行Python脚本并获取返回值 java运行Python脚本并获取返回值 使用runtime类实现在java中运行Python脚本,话不多说,直接上代码 java运行Python脚本并获取返 ...

  9. python执行cmd系统命令获取返回值的几种方式

    python 执行shell命令 注:python3.X模块中以废弃commands该模块,使用subprocess模块在本文下面增加解说 本篇介绍如何使用python在cmd上执行shell命令 用 ...

最新文章

  1. 任正非:华为5G芯片用在iPhone上?我持开放态度
  2. java:不同数据类型的转换规则
  3. 状态管理技术~~Session
  4. 【Elasticsearch】我在 Elasticsearch 集群内应该设置多少个分片?
  5. 谷歌母公司2019Q1财报出炉,营收增速放缓,盘后大跌7.34%,市值蒸发700亿刀
  6. silverlight 3 blend3最新版本 破解方法
  7. editplus怎么在前后插入字符
  8. Python批处理文件
  9. Google Go TCP 开发
  10. 计算机del键作用,计算机里的英文字母“DEL”键是干什么用的
  11. PAT1003甲级 java
  12. javascript(JS) 0基础快速入门 (二)(this指向问题)
  13. 可修改UID的白卡,请问为什么一般读写器不能修改这种卡的UID,必须要特定的读写器才能修改?
  14. jacob操作word教程
  15. ESC32硬件---电路分析二(电流检测和反电动势检测)
  16. CPU上的导热硅脂和散热片大比拼
  17. python独立网站教程_python从零到独立建设网站系列教程之基础知识篇(二)
  18. 如何计算心跳c语言编程,单片机心率计 电子脉搏计设计(原理图Protues仿真和C程序)...
  19. Java + 腾讯企业邮箱 + javamail发送邮件
  20. 关于toc约束理论的学习与体会

热门文章

  1. 第十六届全国大学生智能车竞赛线上比赛-赛前开题
  2. 这些赛道设计的点子,不让用太可惜了 | 赛道审核环节
  3. 英飞凌AI越野组入门教程
  4. 在国内安装Pytorchy以及遇到的问题
  5. Python | [a for b in c for a in b]的用法
  6. php负载均衡原理_PHP超级负载均衡
  7. 字典生成_Python数据字典生成工具详解
  8. android heic图片,如何在Windows / MacOS / Android上打開HEIC照片
  9. oracle11g怎么显示中文,ORACLE11G中PLSQL中文显示乱码、Linux下sqlplus查询中文乱码
  10. java对象的状态,Java编程菜鸟课程:状态对象