Python Language
Écrire des extensions
Recherche…
Bonjour tout le monde avec l'extension C
Le fichier source C suivant (que nous appellerons hello.c
à des fins de démonstration) produit un module d'extension nommé hello
qui contient une seule fonction greet()
:
#include <Python.h>
#include <stdio.h>
#if PY_MAJOR_VERSION >= 3
#define IS_PY3K
#endif
static PyObject *hello_greet(PyObject *self, PyObject *args)
{
const char *input;
if (!PyArg_ParseTuple(args, "s", &input)) {
return NULL;
}
printf("%s", input);
Py_RETURN_NONE;
}
static PyMethodDef HelloMethods[] = {
{ "greet", hello_greet, METH_VARARGS, "Greet the user" },
{ NULL, NULL, 0, NULL }
};
#ifdef IS_PY3K
static struct PyModuleDef hellomodule = {
PyModuleDef_HEAD_INIT, "hello", NULL, -1, HelloMethods
};
PyMODINIT_FUNC PyInit_hello(void)
{
return PyModule_Create(&hellomodule);
}
#else
PyMODINIT_FUNC inithello(void)
{
(void) Py_InitModule("hello", HelloMethods);
}
#endif
Pour compiler le fichier avec le compilateur gcc
, exécutez la commande suivante dans votre terminal préféré:
gcc /path/to/your/file/hello.c -o /path/to/your/file/hello
Pour exécuter la fonction greet()
que nous avons écrite précédemment, créez un fichier dans le même répertoire et appelez-le hello.py
import hello # imports the compiled library
hello.greet("Hello!") # runs the greet() function with "Hello!" as an argument
Passer un fichier ouvert à C Extensions
Transmettez un objet fichier ouvert de Python au code d'extension C.
Vous pouvez convertir le fichier en un descripteur de fichier entier à l'aide de la fonction PyObject_AsFileDescriptor
:
PyObject *fobj;
int fd = PyObject_AsFileDescriptor(fobj);
if (fd < 0){
return NULL;
}
Pour reconvertir un descripteur de fichier entier en objet python, utilisez PyFile_FromFd
.
int fd; /* Existing file descriptor */
PyObject *fobj = PyFile_FromFd(fd, "filename","r",-1,NULL,NULL,NULL,1);
Extension C utilisant c ++ et Boost
Ceci est un exemple de base d'une extension C utilisant C ++ et Boost .
Code C ++
Code C ++ placé dans hello.cpp:
#include <boost/python/module.hpp>
#include <boost/python/list.hpp>
#include <boost/python/class.hpp>
#include <boost/python/def.hpp>
// Return a hello world string.
std::string get_hello_function()
{
return "Hello world!";
}
// hello class that can return a list of count hello world strings.
class hello_class
{
public:
// Taking the greeting message in the constructor.
hello_class(std::string message) : _message(message) {}
// Returns the message count times in a python list.
boost::python::list as_list(int count)
{
boost::python::list res;
for (int i = 0; i < count; ++i) {
res.append(_message);
}
return res;
}
private:
std::string _message;
};
// Defining a python module naming it to "hello".
BOOST_PYTHON_MODULE(hello)
{
// Here you declare what functions and classes that should be exposed on the module.
// The get_hello_function exposed to python as a function.
boost::python::def("get_hello", get_hello_function);
// The hello_class exposed to python as a class.
boost::python::class_<hello_class>("Hello", boost::python::init<std::string>())
.def("as_list", &hello_class::as_list)
;
}
Pour compiler cela dans un module python, vous aurez besoin des en-têtes python et des bibliothèques boost. Cet exemple a été réalisé sur Ubuntu 12.04 en utilisant python 3.4 et gcc. Boost est pris en charge sur de nombreuses plates-formes. Dans le cas d'Ubuntu, les paquets nécessaires ont été installés en utilisant:
sudo apt-get install gcc libboost-dev libpython3.4-dev
Compiler le fichier source dans un fichier .so pouvant être importé ultérieurement en tant que module, à condition que ce soit sur le chemin Python:
gcc -shared -o hello.so -fPIC -I/usr/include/python3.4 hello.cpp -lboost_python-py34 -lboost_system -l:libpython3.4m.so
Le code python dans le fichier example.py:
import hello
print(hello.get_hello())
h = hello.Hello("World hello!")
print(h.as_list(3))
Ensuite, python3 example.py
donnera la sortie suivante:
Hello world!
['World hello!', 'World hello!', 'World hello!']