作者博客:bot7.cc
如果你的项目只是有一个相对简单的代码,并不设计复杂的附加库的话,直接使用SWIG转化编译就可以使用了。具体教程请参考:C/C++转化Python扩展 - 简单例子。
但是,有很多时候我们的项目会相对复杂一些,这时候如果能用IDE进行编译链接就会省事很多。像我的项目中用到了Qt的库,而SWIG本身并不支持Qt的库。解决的思路就是在外层再做一层封装,只用标准库来编写入口的类文件,只是使用SWIG转化最外层的这个类就可以了。
用Qt Creator创建待转化的项目
新建C/C++ Library项目Example,选择静态库。
添加一个简单的类作为示例:
// example.h
#ifndef EXAMPLE_H
#define EXAMPLE_H
#include <iostream>
#include <QString>
class Example
{
private:
QString *m_str;
public:
Example();
void sayHello();
};
#endif // EXAMPLE_H
// example.cpp
#include "example.h"
using namespace std;
Example::Example()
{
m_str = new QString("Hello Python, we are Qt and C++!");
}
void Example::sayHello(){
cout << m_str->toLatin1().data() << endl;
}
在Example这个类中我们用到了Qt的库。
选择Release进行编译,会生成相应的库文件, win下为.lib
,linux下为.a
。
新建一个项目用来封装Example类
利用Qt Creator新建一个C++ Library项目Warp, 选择动态库。
将上一步的example.h
文件和生成的libExample.a
(win下名字不同)文件复制到本项目目录的lib文件夹中。
在.pro
文件中添加包含路径和库:
INCLUDEPATH += lib \
"YOURPYTHONINCLUDEPATH"
LIBS += -Llib -lExample \
-L"YOURPYTHONINCLUDEPATH" -lpythonx.x # 根据你的Python版本配置
添加Warp类:
// warp.h
#ifndef WARP_H
#define WARP_H
class Example;
class Warp
{
private:
Example *m_example;
public:
Warp();
void sayHello();
};
#endif // WARP_H
// warp.cpp
#include "warp.h"
#include "example.h"
Warp::Warp()
{
m_example = new Example();
}
void Warp::sayHello(){
m_example->sayHello();
}
利用SWIG进行转化
编写warp.i文件:
%module warp
%{
#include "warp.h"
%}
%include "warp.h"
运行SWIG生成转化文件:
swig -c++ -python warp.i
之后会生成warp.py
文件和warp_warp.cxx
文件。
将warp_warp.cxx
文件添加到Warp项目当中,选择Release进行编译,生成动态链接库。
将生成的动态链接库改名为_warp.so
(win下后缀不同)。
此时即生成了所有所需的文件,warp.py
, _warp.so
。
在Python中调用
此时已经可以在上一步编译输出的目录下使用Python扩展了,使用import warp
导入即可。