- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章如何在C++中调用Python由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
为了使用Python.h这个扩展项,我们需要安装一个python*-dev而不是python*,这两者略有区别,下面的案例展示的是在Ubuntu20.04下安装python3.9-dev的方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
|
dechin@ubuntu2004:~
/projects/gitlab/dechin/
$
sudo
apt
install
python3.9-dev
正在读取软件包列表... 完成
正在分析软件包的依赖关系树
正在读取状态信息... 完成
下列软件包是自动安装的并且现在不需要了:
chromium-codecs-ffmpeg-extra gstreamer1.0-vaapi
libgstreamer-plugins-bad1.0-0 linux-headers-5.8.0-43-generic
linux-hwe-5.8-headers-5.8.0-43 linux-image-5.8.0-43-generic
linux-modules-5.8.0-43-generic linux-modules-extra-5.8.0-43-generic
使用
'sudo apt autoremove'
来卸载它(它们)。
将会同时安装下列软件:
libexpat1-dev libpython3.9 libpython3.9-dev zlib1g-dev
下列【新】软件包将被安装:
libexpat1-dev libpython3.9 libpython3.9-dev python3.9-dev zlib1g-dev
升级了 0 个软件包,新安装了 5 个软件包,要卸载 0 个软件包,有 30 个软件包未被升级。
需要下载 6,613 kB 的归档。
解压缩后会消耗 28.7 MB 的额外空间。
您希望继续执行吗? [Y
/n
] Y
获取:1 http:
//repo
.huaweicloud.com
/ubuntu
focal
/main
amd64 libexpat1-dev amd64 2.2.9-1build1 [116 kB]
获取:2 http:
//repo
.huaweicloud.com
/ubuntu
focal-updates
/universe
amd64 libpython3.9 amd64 3.9.0-5~20.04 [1,710 kB]
获取:3 http:
//repo
.huaweicloud.com
/ubuntu
focal-updates
/universe
amd64 libpython3.9-dev amd64 3.9.0-5~20.04 [4,119 kB]
获取:4 http:
//repo
.huaweicloud.com
/ubuntu
focal-updates
/main
amd64 zlib1g-dev amd64 1:1.2.11.dfsg-2ubuntu1.2 [155 kB]
获取:5 http:
//repo
.huaweicloud.com
/ubuntu
focal-updates
/universe
amd64 python3.9-dev amd64 3.9.0-5~20.04 [512 kB]
已下载 6,613 kB,耗时 4秒 (1,594 kB
/s
)
正在选中未选择的软件包 libexpat1-dev:amd64。
(正在读取数据库 ... 系统当前共安装有 269544 个文件和目录。)
准备解压 ...
/libexpat1-dev_2
.2.9-1build1_amd64.deb ...
正在解压 libexpat1-dev:amd64 (2.2.9-1build1) ...
正在选中未选择的软件包 libpython3.9:amd64。
准备解压 ...
/libpython3
.9_3.9.0-5~20.04_amd64.deb ...
正在解压 libpython3.9:amd64 (3.9.0-5~20.04) ...
正在选中未选择的软件包 libpython3.9-dev:amd64。
准备解压 ...
/libpython3
.9-dev_3.9.0-5~20.04_amd64.deb ...
正在解压 libpython3.9-dev:amd64 (3.9.0-5~20.04) ...
正在选中未选择的软件包 zlib1g-dev:amd64。
准备解压 ...
/zlib1g-dev_1
%3a1.2.11.dfsg-2ubuntu1.2_amd64.deb ...
正在解压 zlib1g-dev:amd64 (1:1.2.11.dfsg-2ubuntu1.2) ...
正在选中未选择的软件包 python3.9-dev。
准备解压 ...
/python3
.9-dev_3.9.0-5~20.04_amd64.deb ...
正在解压 python3.9-dev (3.9.0-5~20.04) ...
正在设置 libpython3.9:amd64 (3.9.0-5~20.04) ...
正在设置 libexpat1-dev:amd64 (2.2.9-1build1) ...
正在设置 zlib1g-dev:amd64 (1:1.2.11.dfsg-2ubuntu1.2) ...
正在设置 libpython3.9-dev:amd64 (3.9.0-5~20.04) ...
正在设置 python3.9-dev (3.9.0-5~20.04) ...
正在处理用于
man
-db (2.9.1-1) 的触发器 ...
正在处理用于 libc-bin (2.31-0ubuntu9.2) 的触发器 ...
|
安装完成后,如果在当前命令行下运行python3.9,是可以看到一个python专属的命令行界面的,可以通过exit()退出。但是我们这里侧重的是跟C++的配合工作,因此我们更加关注lib和include目录下是否有生成相关的目录,可以执行如下指令进行查看:
1
2
3
4
5
|
dechin@ubuntu2004:~
/projects/gitlab/dechin/
$ ll
/usr/lib/
|
grep
python
drwxr-xr-x 26 root root 20480 5月 7 16:27 python2.7/
drwxr-xr-x 3 root root 4096 2月 10 02:47 python3/
drwxr-xr-x 30 root root 20480 5月 7 16:30 python3.8/
drwxr-xr-x 31 root root 12288 5月 20 16:31 python3.9/
|
这里我们看到有一个3.9的版本,也就是我们刚才安装的版本,再看看include下的目录:
1
2
3
|
dechin@ubuntu2004:~
/projects/gitlab/dechin/
$ ll
/usr/include/
|
grep
python
drwxr-xr-x 2 root root 4096 5月 7 16:31 python3.8/
drwxr-xr-x 4 root root 4096 5月 20 16:31 python3.9/
|
这里我们就可以看到一些区别了,有一些版本的python不一定会有这两个目录,但是只有具备了这两个目录,才能够被C++调用.
这里我们使用的IDE是VS Code,但是上述提到的几个路径,在VS Code中默认是不被包含的,因此在代码编辑的过程中在include <Python.h>这一步就会报错了。这一章节的目的主要是解决IDE中的报错问题,还不是最终运行中出现的问题,因为运行时我是通过命令行执行g++来运行的,而不是直接用IDE来跑。首先在VS Code界面上按顺序同时按住:ctrl+shift+P,在弹出的窗口中输入C/C++ Edit Configurations(JSON)查找相关JSON配置文件,在列表中点击后会自动在VS Code中打开这个配置文件:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
{
"configurations"
: [
{
"name"
:
"Linux"
,
"includePath"
: [
"${workspaceFolder}/**"
],
"defines"
: [],
"compilerPath"
:
"/usr/bin/gcc"
,
"cStandard"
:
"gnu17"
,
"cppStandard"
:
"c++11"
,
"intelliSenseMode"
:
"linux-gcc-x64"
}
],
"version"
:
4
}
|
我们所需要做的工作就是,在这个includePath中把相关的路径都加上,比如我这边添加的路径是以下3个:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
{
"configurations"
: [
{
"name"
:
"Linux"
,
"includePath"
: [
"${workspaceFolder}/**"
,
"/usr/include/python3.9/"
,
"/usr/lib/python3.9/"
,
"/usr/include/python3.9/cpython/"
],
"defines"
: [],
"compilerPath"
:
"/usr/bin/gcc"
,
"cStandard"
:
"gnu17"
,
"cppStandard"
:
"c++11"
,
"intelliSenseMode"
:
"linux-gcc-x64"
}
],
"version"
:
4
}
|
添加后,include <Python.h>就不会显示报错了.
行业潜规则,我们先用C++来调用一个Python的打印函数,输出Hello World试试:
1
2
3
4
5
6
7
8
|
// cp.cpp
#include <Python.h>
int
main(
int
argc,
char
*argv[]) {
Py_Initialize();
PyRun_SimpleString(
"print('hello world')\n"
);
Py_Finalize();
return
0;
}
|
这里需要注意的是一个运行方式,我们是用g++来进行编译的,但是g++默认是找不到我们刚才在IDE中所设定的几个includePath的,因此需要我们手动在编译的时候加上几个参数。这些参数其实也可以运行python3.9-config去一个一个查看,这里我们直接推荐一种可以运行成功的参数,其中最重要的是-I和-l这两个路径一定要包含:
1
2
3
4
5
6
7
|
dechin@ubuntu2004:~
/projects/gitlab/dechin/
$ g++ -o cpy
cp
.cpp -lm -std=c++11 -I
/usr/include/python3
.9/ -lpython3.9
dechin@ubuntu2004:~
/projects/gitlab/dechin/
$ ll
总用量 4697388
drwxrwxr-x 2 dechin dechin 4096 5月 20 17:10 ./
drwxrwxr-x 8 dechin dechin 4096 5月 19 15:32 ../
-rw-rw-r-- 1 dechin dechin 152 5月 20 17:04
cp
.cpp
-rwxrwxr-x 1 dechin dechin 16776 5月 20 17:10 cpy*
|
运行完成后,就会在当前目录下生成一个刚才指定的名字cpy的一个可执行文件,如果是windows系统,则会生成一个cpy.exe的文件。让我们执行这个文件:
1
2
|
dechin@ubuntu2004:~
/projects/gitlab/dechin/
$ .
/cpy
hello world
|
成功打印Hello World,又离成功更近了一步.
在C++中如果我们想分割一个字符串,虽然说也是可以实现的,但是应该没有比Python中执行一个string.split()更加方便快捷的方案了,因此我们测试一个用C++调用Python的split函数的功能.
一开始我们是写了这样一个简单的案例,用PyImport_ImportModule方法去调用pysplit这个python模块:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
// cp.cpp
#include <Python.h>
#include <iostream>
using
namespace
std;
int
main(
int
argc,
char
*argv[])
{
Py_Initialize();
if
(!Py_IsInitialized())
{
cout <<
"Initialize failed!"
<< endl;
return
0;
}
PyObject* pModule = NULL;
PyObject* pFunc;
PyRun_SimpleString(
"import os"
);
PyRun_SimpleString(
"os.system('pwd')"
);
pModule = PyImport_ImportModule(
"pysplit"
);
if
(pModule == NULL)
{
cout <<
"Module Not Found!"
<< endl;
}
// pFunc = PyObject_GetAttrString(pModule, "sp");
// PyObject* args = Py_BuildValue("s", "Test String Hello Every One !");
// PyObject* pRet = PyObject_CallObject(pFunc, args);
string cList[10];
// PyArg_Parse(pRet, "[items]", &cList);
cout <<
"res:"
<< cList << endl;
Py_Finalize();
return
0;
}
|
对应的Python模块的内容为:
1
2
3
4
|
# pysplit.py
def
sp(string):
return
string.split()
|
这是一个非常简单的函数,但是我们在调用的时候就直接返回了一个错误:
1
2
3
4
|
dechin@ubuntu2004:~
/
projects
/
gitlab
/
dechin
/
$ g
+
+
-
o cpy cp.cpp
-
lm
-
std
=
c
+
+
11
-
I
/
usr
/
include
/
python3.
9
/
-
lpython3.
9
&& .
/
cpy
[
'pysplit.py'
,
'cpy'
,
'cp.cpp'
]
Module Not Found!
res:
0x7ffc622ae900
|
这个错误是说,找不到pysplit这个模块。但是我们同时借助于PyRun_SimpleString调用了Python中的os库,执行了一个查看路径和当前路径下文件的功能,我们发现这个C++文件和需要引入的pysplit.py其实是在同一个路径下的,这就很奇怪了没有导入成功.
经过一番的资料查询,最后发现,即使是在相同的路径下,也需要通过Python的sys将当前目录添加到系统路径中,才能够识别到这个模块,同样也是使用PyRun_SimpleString的函数:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
/
/
cp.cpp
#include <Python.h>
#include <iostream>
using namespace std;
int
main(
int
argc, char
*
argv[])
{
Py_Initialize();
if
(!Py_IsInitialized())
{
cout <<
"Initialize failed!"
<< endl;
return
0
;
}
PyObject
*
pModule
=
NULL;
PyObject
*
pFunc;
PyRun_SimpleString(
"import sys"
);
PyRun_SimpleString(
"sys.path.append('./')"
);
pModule
=
PyImport_ImportModule(
"pysplit"
);
if
(pModule
=
=
NULL)
{
cout <<
"Module Not Found!"
<< endl;
}
pFunc
=
PyObject_GetAttrString(pModule,
"sp"
);
PyObject
*
args
=
Py_BuildValue(
"s"
,
"Test String Hello Every One !"
);
PyObject
*
pRet
=
PyObject_CallObject(pFunc, args);
string cList[
10
];
/
/
PyArg_Parse(pRet,
"[items]"
, &cList);
cout <<
"res:"
<< cList << endl;
Py_Finalize();
return
0
;
}
|
这个也可以理解,Python中的函数调用,输入参数都被打包成了一个tuple格式,比如**args,而类似**kwargs则是打包成一个字典格式,类似的功能在这篇博客中有所介绍.
上面的问题,在StackOverFlow上有一个类似的情况,有一个回答解决了这个问题,解决方案是,用PyObject_CallFunctionObjArgs来替代PyObject_CallObject去实现函数调用命令,相关代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
// cp.cpp
#include <Python.h>
#include <iostream>
using
namespace
std;
int
main(
int
argc,
char
*argv[])
{
Py_Initialize();
if
(!Py_IsInitialized())
{
cout <<
"Initialize failed!"
<< endl;
return
0;
}
PyObject* pModule = NULL;
PyObject* pFunc;
PyRun_SimpleString(
"import sys"
);
PyRun_SimpleString(
"sys.path.append('./')"
);
pModule = PyImport_ImportModule(
"pysplit"
);
if
(pModule == NULL)
{
cout <<
"Module Not Found!"
<< endl;
}
pFunc = PyObject_GetAttrString(pModule,
"sp"
);
PyObject* args = Py_BuildValue(
"s"
,
"Test String Hello Every One !"
);
PyObject* pRet = PyObject_CallFunctionObjArgs(pFunc, args, NULL);
int
size = PyList_Size(pRet);
cout <<
"List size is: "
<< size << endl;
for
(
int
i=0;i<size;i++)
{
PyObject* cRet = PyList_GET_ITEM(pRet, i);
char
* s;
PyArg_Parse(cRet,
"s"
, &s);
cout <<
"The "
<< i <<
"th term is: "
<< s << endl;
}
Py_Finalize();
return
0;
}
|
最后,因为从Python中获取的是一个List格式的数据,因此我们首先需要用PyList_GET_ITEM去逐项提取,然后用PyArg_Parse将提取出来的元素保存到一个C++的char字符串中,执行结果如下:
1
2
3
4
5
6
7
8
|
dechin@ubuntu2004:~
/projects/gitlab/dechin/
$ g++ -o cpy
cp
.cpp -lm -std=c++11 -I
/usr/include/python3
.9/ -lpython3.9 && .
/cpy
List size is: 6
The 0th term is: Test
The 1th term is: String
The 2th term is: Hello
The 3th term is: Every
The 4th term is: One
The 5th term is: !
|
Yes!终于成功了! 。
本文介绍了一个在C++内部调用Python中封装的函数或者接口的方法,从环境配置到具体示例都有讲解,并且在其中包含有不少的坑点,需要一步一步去踩。不同的编程语言具有不同的优势,Python轮子众多而语法简单,上手容易,但是性能比较首先,C++的最明显优势就是在于其性能的天然优越性。但是我们不需要对哪一种编程语言有所偏倚,都有所掌握,并且能够有所互通,利用好各自的优势,才能够发挥最大的价值.
以上就是如何在C++中调用Python的详细内容,更多关于C++ 调用Python的资料请关注我其它相关文章! 。
原文链接:https://www.cnblogs.com/dechinphy/p/cpp-python.html 。
最后此篇关于如何在C++中调用Python的文章就讲到这里了,如果你想了解更多关于如何在C++中调用Python的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我正在处理一组标记为 160 个组的 173k 点。我想通过合并最接近的(到 9 或 10 个组)来减少组/集群的数量。我搜索过 sklearn 或类似的库,但没有成功。 我猜它只是通过 knn 聚类
我有一个扁平数字列表,这些数字逻辑上以 3 为一组,其中每个三元组是 (number, __ignored, flag[0 or 1]),例如: [7,56,1, 8,0,0, 2,0,0, 6,1,
我正在使用 pipenv 来管理我的包。我想编写一个 python 脚本来调用另一个使用不同虚拟环境(VE)的 python 脚本。 如何运行使用 VE1 的 python 脚本 1 并调用另一个 p
假设我有一个文件 script.py 位于 path = "foo/bar/script.py"。我正在寻找一种在 Python 中通过函数 execute_script() 从我的主要 Python
这听起来像是谜语或笑话,但实际上我还没有找到这个问题的答案。 问题到底是什么? 我想运行 2 个脚本。在第一个脚本中,我调用另一个脚本,但我希望它们继续并行,而不是在两个单独的线程中。主要是我不希望第
我有一个带有 python 2.5.5 的软件。我想发送一个命令,该命令将在 python 2.7.5 中启动一个脚本,然后继续执行该脚本。 我试过用 #!python2.7.5 和http://re
我在 python 命令行(使用 python 2.7)中,并尝试运行 Python 脚本。我的操作系统是 Windows 7。我已将我的目录设置为包含我所有脚本的文件夹,使用: os.chdir("
剧透:部分解决(见最后)。 以下是使用 Python 嵌入的代码示例: #include int main(int argc, char** argv) { Py_SetPythonHome
假设我有以下列表,对应于及时的股票价格: prices = [1, 3, 7, 10, 9, 8, 5, 3, 6, 8, 12, 9, 6, 10, 13, 8, 4, 11] 我想确定以下总体上最
所以我试图在选择某个单选按钮时更改此框架的背景。 我的框架位于一个类中,并且单选按钮的功能位于该类之外。 (这样我就可以在所有其他框架上调用它们。) 问题是每当我选择单选按钮时都会出现以下错误: co
我正在尝试将字符串与 python 中的正则表达式进行比较,如下所示, #!/usr/bin/env python3 import re str1 = "Expecting property name
考虑以下原型(prototype) Boost.Python 模块,该模块从单独的 C++ 头文件中引入类“D”。 /* file: a/b.cpp */ BOOST_PYTHON_MODULE(c)
如何编写一个程序来“识别函数调用的行号?” python 检查模块提供了定位行号的选项,但是, def di(): return inspect.currentframe().f_back.f_l
我已经使用 macports 安装了 Python 2.7,并且由于我的 $PATH 变量,这就是我输入 $ python 时得到的变量。然而,virtualenv 默认使用 Python 2.6,除
我只想问如何加快 python 上的 re.search 速度。 我有一个很长的字符串行,长度为 176861(即带有一些符号的字母数字字符),我使用此函数测试了该行以进行研究: def getExe
list1= [u'%app%%General%%Council%', u'%people%', u'%people%%Regional%%Council%%Mandate%', u'%ppp%%Ge
这个问题在这里已经有了答案: Is it Pythonic to use list comprehensions for just side effects? (7 个答案) 关闭 4 个月前。 告
我想用 Python 将两个列表组合成一个列表,方法如下: a = [1,1,1,2,2,2,3,3,3,3] b= ["Sun", "is", "bright", "June","and" ,"Ju
我正在运行带有最新 Boost 发行版 (1.55.0) 的 Mac OS X 10.8.4 (Darwin 12.4.0)。我正在按照说明 here构建包含在我的发行版中的教程 Boost-Pyth
学习 Python,我正在尝试制作一个没有任何第 3 方库的网络抓取工具,这样过程对我来说并没有简化,而且我知道我在做什么。我浏览了一些在线资源,但所有这些都让我对某些事情感到困惑。 html 看起来
我是一名优秀的程序员,十分优秀!