Dadurch wird Python nicht zu Maschinencode kompiliert. Ermöglicht jedoch das Erstellen einer gemeinsam genutzten Bibliothek zum Aufrufen von Python-Code.
Wenn Sie nach einer einfachen Möglichkeit suchen, Python-Code von C aus auszuführen, ohne sich auf Execp-Inhalte verlassen zu müssen. Sie können eine gemeinsam genutzte Bibliothek aus Python-Code generieren, der mit einigen Aufrufen der Python-Einbettungs-API umschlossen ist . Nun, die Anwendung ist eine gemeinsam genutzte Bibliothek, eine .so, die Sie in vielen anderen Bibliotheken / Anwendungen verwenden können.
Hier ist ein einfaches Beispiel, das eine gemeinsam genutzte Bibliothek erstellt, die Sie mit einem C-Programm verknüpfen können. Die gemeinsam genutzte Bibliothek führt Python-Code aus.
Die Python-Datei, die ausgeführt wird, lautet pythoncalledfromc.py
:
# -*- encoding:utf-8 -*-
# this file must be named "pythoncalledfrom.py"
def main(string): # args must a string
print "python is called from c"
print "string sent by «c» code is:"
print string
print "end of «c» code input"
return 0xc0c4 # return something
Sie können es mit versuchen python2 -c "import pythoncalledfromc; pythoncalledfromc.main('HELLO')
. Es wird ausgegeben:
python is called from c
string sent by «c» code is:
HELLO
end of «c» code input
Die gemeinsam genutzte Bibliothek wird wie folgt definiert callpython.h
:
#ifndef CALL_PYTHON
#define CALL_PYTHON
void callpython_init(void);
int callpython(char ** arguments);
void callpython_finalize(void);
#endif
Das zugehörige callpython.c
ist:
// gcc `python2.7-config --ldflags` `python2.7-config --cflags` callpython.c -lpython2.7 -shared -fPIC -o callpython.so
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <python2.7/Python.h>
#include "callpython.h"
#define PYTHON_EXEC_STRING_LENGTH 52
#define PYTHON_EXEC_STRING "import pythoncalledfromc; pythoncalledfromc.main(\"%s\")"
void callpython_init(void) {
Py_Initialize();
}
int callpython(char ** arguments) {
int arguments_string_size = (int) strlen(*arguments);
char * python_script_to_execute = malloc(arguments_string_size + PYTHON_EXEC_STRING_LENGTH);
PyObject *__main__, *locals;
PyObject * result = NULL;
if (python_script_to_execute == NULL)
return -1;
__main__ = PyImport_AddModule("__main__");
if (__main__ == NULL)
return -1;
locals = PyModule_GetDict(__main__);
sprintf(python_script_to_execute, PYTHON_EXEC_STRING, *arguments);
result = PyRun_String(python_script_to_execute, Py_file_input, locals, locals);
if(result == NULL)
return -1;
return 0;
}
void callpython_finalize(void) {
Py_Finalize();
}
Sie können es mit dem folgenden Befehl kompilieren:
gcc `python2.7-config --ldflags` `python2.7-config --cflags` callpython.c -lpython2.7 -shared -fPIC -o callpython.so
Erstellen Sie eine Datei mit dem Namen callpythonfromc.c
, die Folgendes enthält:
#include "callpython.h"
int main(void) {
char * example = "HELLO";
callpython_init();
callpython(&example);
callpython_finalize();
return 0;
}
Kompilieren Sie es und führen Sie Folgendes aus:
gcc callpythonfromc.c callpython.so -o callpythonfromc
PYTHONPATH=`pwd` LD_LIBRARY_PATH=`pwd` ./callpythonfromc
Dies ist ein sehr einfaches Beispiel. Es kann funktionieren, aber je nach Bibliothek kann es immer noch schwierig sein, C-Datenstrukturen nach Python und von Python nach C zu serialisieren. Die Dinge können etwas automatisiert werden ...
Nuitka könnte hilfreich sein.
Es gibt auch Numba, aber beide wollen nicht genau das tun, was Sie wollen. Das Generieren eines C-Headers aus Python-Code ist möglich, jedoch nur, wenn Sie angeben, wie die Python-Typen in C-Typen konvertiert werden sollen, oder auf diese Informationen schließen können. Siehe Python Astroid für einen Python Ast Analyzer.