时间: 2020-09-4|tag: 45次围观|0 条评论


一、背景

今年3月份辞职,进入了一家新公司,公司是做网络存储和数据智能的。进去不到一个月,老大给我叫我把ipmi集成到项目中。

以前一直未做过将开源项目集成到项目中,对自己来说是一种挑战。

二、探索

ipmi是inter出的一个协议,用于监测服务器硬件状态等,如电源、CPU、风扇、温度等

刚接受任务,满怀信心开始做,首先先了解一下ipmi是啥东东。

之后发现现在有很多ipmi工具已经封装了ipmi,即:ipmitool,ipmiutils、freeipmi,OpenIPMI等,直接使用工具就可以监测硬件。

于是,我就开始安装了ipmitool,然后用C的系统函数system调用ipmitool命令,返回一些结果,然后对结果进行处理。

其实这样很简单,当我把demo拿个老大看得时候。老大说可以这样做,但是这样做不好,直接封装api更好,其实ipmitool也是封装api做的。

所以,我就开始尝试去看api,看能不能封装api。

于是乎,漫长的痛苦之旅启程了。完全看不懂,而且资料少之又少。最重要是不知道程序处理流程是啥。各位博友,如果有人做过这方面(开源代码封装)的开发,能否详细交流一下,互相学习,感激不尽。我这个人技术虽然不是太好,普通的程序猿,但是我很爱这行,爱交知心朋友,希望各位同行,多多联系,多多交流学习。

我就给老大说这个搞不定,老大说要不你就用命令吧,用Python写最好。

当听到python,感觉还是很友好的,因为之前写过一些python,感觉还是很不错。

于是我就尝试python封装ipmitool命令,并用c去调用python。可能咋一看,各位会感觉多此一举,python调用c,c又调用python。

之所以用python,是因为,python正则处理非常强大。

三、难题

我简单看了一下c调用python的代码(网上很多),就写了一个demo,用c调用python类中的函数,运行起来,答应了一条hello world,感觉还不错。

下面c_python_utils.h是处理工具函数,test.cpp是测试程序,hello.py是python类

可是当我集成到项目中的时候,PyImport_Import总是返回为空起初我以为是init()中设置目录的问题,但是目录设置的是当前目录,hello.py也放在当前目录。

然而,然后让我发现项目是部署在服务器上的,而服务器上可执行文件在某个目录内,这个目录下根本就没有hello.py(因为我没有放进去)

所以,根本就在当前目录下找不到,就是这个小小的问题,都把我折腾了好久,现在想起来真是心伤啊。不过还是解决了。

/***************************************************************************************************c_python_utils.h    C++ Network Library, Copyright (c) Datatom Software, Inc.(2015)Author:    liu.pan (liu.pan@datatom.com)    Creating Time:    2015-5-4***************************************************************************************************/#ifndef _DTCORE_C_PYTHON_UTILS_H_#define _DTCORE_C_PYTHON_UTILS_H_#include <Python.h>#include <stdio.h>#ifdef __cplusplusextern "C" {#endif /* C++ *//*** 调用python类中的成员函数* @param module python脚本名称,不含扩展* @param class_name python类名称* @param function python类成员函数* @param format python类函数参数格式* @return 返回字符串*/char* py_call( const char* module, const char* class_name, char* function, char* format, ... ){    PyObject* pName     = NULL;    PyObject* pMod         = NULL;    PyObject* pDict     = NULL;    PyObject* pClass    = NULL;    PyObject* pInstance = NULL;    PyObject* pParam     = NULL;    PyObject* pResult     = NULL;    // 导入模块    pName = PyString_FromString(module);    pMod = PyImport_Import(pName);    if( !pMod )    {        return "";    }    // 获取模块字典属性    pDict = PyModule_GetDict(pMod);    if ( !pDict )    {        return "";    }    // 通过字典获取模块中的类    pClass = PyDict_GetItemString(pDict, class_name);    if ( !pClass )    {        return "";    }    pInstance = PyInstance_New(pClass, NULL, NULL);    if ( !pInstance )    {        return "";    }    pResult = PyObject_CallMethod(pInstance, function, format);        char *rlt_ch = NULL;    PyArg_Parse( pResult, "s", &rlt_ch );    return rlt_ch;}/*** 一些环境的初始化* */void init(){    Py_Initialize();    PyRun_SimpleString("import sys");    PyRun_SimpleString("sys.path.append('./')");}/*** 逆初始化*/void finit(){    Py_Finalize();}#ifdef __cplusplus} /* extern "C" */#endif /* C++ */#endif  //_DTCORE_C_PYTHON_UTILS_H_
// test.cpp#include "c_python_utils.h"#include <stdio.h>int main(int argc, char const *argv[]){    init();    char* rlt_char = py_call("hello", "power", "liupan", "()");    finit();    printf("%s\n", rlt_char);    return 0;}
# hello.pyclass power():    def liupan(self):        return "hello world"

四、总结

所以,如果PyImport_Import总是返回为空,一定是查询目录的问题,要么目录设置错误,要么python代码根本没有在这个目录内。

因为C调用python是运行时执行的,而不是编译时链接的。

文章转载于:https://www.cnblogs.com/panliu/p/4485137.html

原著是一个有趣的人,若有侵权,请通知删除

本博客所有文章如无特别注明均为原创。
复制或转载请以超链接形式注明转自起风了,原文地址《参加新工作以来的一些吐槽
   

还没有人抢沙发呢~