
在现代软件开发中,不同编程语言的优势互补已成为常态。go语言以其出色的并发能力和高性能在后端服务领域占据一席之地,而python则凭借其丰富的库生态和开发效率在数据科学、web开发等领域广受欢迎。然而,如何在python项目中高效地复用go语言编写的功能,是许多开发者面临的挑战。直接调用通常是不可能的,需要通过中间层进行桥接。本文将探讨两种主要的解决方案:swig和cython,它们都利用了go语言编译到c语言的能力作为桥梁。
SWIG(Simplified Wrapper and Interface Generator)是一个强大的工具,能够自动生成多种脚本语言与C/C++代码之间的接口。尽管SWIG的Go示例通常展示的是Go调用C/C++函数,但其核心机制是为C/C++接口生成目标语言的绑定。因此,如果Go函数能够被编译成C语言接口,那么SWIG理论上就可以介入。
实现路径:Go → C (通过cgo) → SWIG → Python
Go代码编译为C兼容库: Go语言的cgo工具允许Go代码与C代码进行互操作。通过cgo,开发者可以将Go函数导出为C语言可调用的形式。这意味着Go编译器(如Go的默认编译器配合cgo指令)可以生成包含C语言ABI(Application Binary Interface)的共享库(如.so或.dll文件),其中包含了Go函数的C语言包装。
// example.go
package main
/*
#include <stdio.h>
extern int add(int a, int b);
*/
import "C"
import "fmt"
//export add
func add(a, b int) int {
return a + b
}
//export greet
func greet(name *C.char) {
fmt.Printf("Hello, %s from Go!\n", C.GoString(name))
}
func main() {} // 需要一个main函数,即使是空的,以便编译为库通过go build -buildmode=c-shared -o libexample.so example.go命令,可以生成一个C共享库libexample.so和一个头文件libexample.h,其中包含了add和greet函数的C语言声明。
立即学习“Python免费学习笔记(深入)”;
SWIG生成Python绑定: 一旦有了C头文件和C共享库,SWIG就可以读取C头文件,并根据其定义生成Python模块,这些模块能够加载并调用C共享库中的函数。
// example.h (由cgo生成或手动编写的C接口定义) extern int add(int a, int b); extern void greet(char* name);
然后,使用SWIG接口文件(.i)来指示SWIG如何生成Python绑定:
// example.i
%module example
%{
#include "example.h"
%}
%include "example.h"接着,通过SWIG命令行工具生成C包装文件和Python模块: swig -python -o example_wrap.c example.igcc -shared -o _example.so example_wrap.c libexample.so -I/usr/include/pythonX.Y -lpythonX.Y (具体命令取决于系统和Python版本)
Python调用: 最后,Python代码就可以像导入普通模块一样导入生成的_example模块,并调用其中的Go函数。
import example
result = example.add(10, 20)
print(f"Go add(10, 20) = {result}")
example.greet("Python User")注意事项:
Cython是一种优化Python代码的语言,它允许开发者用Python语法编写C扩展模块。Cython的强大之处在于它能够将Python代码编译为C代码,并直接与C/C++库进行交互。这为Python调用Go函数提供了一条更直接的途径,尤其是在不希望引入SWIG的情况下。
实现路径:Go → C (通过cgo) → Cython → Python
Go代码编译为C兼容库: 与SWIG方案相同,首先需要使用cgo将Go函数导出为C语言可调用的接口,并生成相应的C头文件和共享库。
Cython封装C函数: Cython可以直接声明和调用C函数。开发者可以在Cython .pxd 文件中声明Go导出的C函数签名,并在 .pyx 文件中编写Python包装器来调用这些C函数。
# my_go_wrapper.pxd (Cython声明文件,对应Go导出的C头文件)
cdef extern from "libexample.h": # 假设cgo生成了libexample.h
int add(int a, int b)
void greet(char* name)
# my_go_wrapper.pyx (Cython实现文件)
def py_add(a: int, b: int) -> int:
"""
通过Cython调用Go导出的C加法函数。
"""
return add(a, b)
def py_greet(name: str):
"""
通过Cython调用Go导出的C问候函数。
"""
# Python字符串需要转换为C的char*
cdef bytes b_name = name.encode('utf-8')
greet(b_name)编译Cython模块: 使用setup.py脚本编译Cython代码为Python可导入的C扩展模块,并链接Go生成的C共享库。
# setup.py
from setuptools import setup, Extension
from Cython.Build import cythonize
extensions = [
Extension(
"my_go_wrapper",
["my_go_wrapper.pyx"],
libraries=["example"], # 链接Go生成的libexample.so
library_dirs=["."], # 指定libexample.so的路径
include_dirs=["."] # 指定libexample.h的路径
)
]
setup(
ext_modules=cythonize(extensions)
)然后运行python setup.py build_ext --inplace进行编译。
Python调用: Python代码可以直接导入my_go_wrapper模块并调用其中的函数。
import my_go_wrapper
result = my_go_wrapper.py_add(15, 25)
print(f"Go add(15, 25) = {result}")
my_go_wrapper.py_greet("Cython User")Cython的“纯C”技术考量: Cython的强大之处在于,通过精心设计,它可以生成不依赖Python运行时(Py_Initialize()等)的“纯C”代码。这意味着生成的C代码可以更加独立,甚至可以被其他C程序直接链接。在封装Go导出的C函数时,如果Cython代码中只使用cdef定义的类型和函数,避免使用Python对象和Python API,那么生成的C代码将更接近原生C,从而减少Python运行时带来的开销和潜在兼容性问题。
这种技术通常涉及:
注意事项:
在Python中调用Go函数并非直接可行,但通过将Go代码编译为C兼容库,并借助SWIG或Cython作为中间桥梁,可以实现这一目标。
无论选择哪种方案,都需要仔细考虑以下几点:
最终,选择哪种方法取决于项目的具体需求、团队的技术栈以及对性能和开发复杂度的权衡。两种方法都提供了可行的路径,但都需要开发者投入时间和精力去理解和实现。
以上就是探索Python调用Go函数的途径:SWIG与Cython的实践考量的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号