Я пытаюсь импортировать модуль в C ++. модуль находится в пакете и должен быть доступен как:
from x.y import class1,func1, const1, etc
Я нахожусь на Python3 .6, и для этой версии я нашел пока использование PyRun_SimpleString
для импорта а затем используйте PyImport_AddModuleObject
, чтобы получить дескриптор моего модуля. то есть:
PyRun_SimpleString("from PacKage1 import Module1 as Module1");
auto module = PyImport_AddModuleObject(PyUnicode_DecodeFSDefault("Module1"));
, чтобы в дальнейшем я мог получить доступ к его различным атрибутам, что-то вроде этого:
auto args = Py_BuildValue("sOOOOONNiN", model_name, model_checkpoint_path, align_fn,
bank_folder_root, cache_folder, postfix,
rebuild_cache, use_jit, threshold, device);
if (module != nullptr) {
// dict is a borrowed reference.
auto pdict = PyModule_GetDict(module);
if (pdict == nullptr) {
cout << "Fails to get the dictionary.\n";
return 1;
}
//Py_DECREF(module);
PyObject *pKeys = PyDict_Keys(pdict);
PyObject *pValues = PyDict_Values(pdict);
map<string, string> my_map;
//cout << "size: " << PyDict_Size(pdict)<<endl;
char* cstr_key = new char[100];
char* cstr_value = new char[500];
for (Py_ssize_t i = 0; i < PyDict_Size(pdict); ++i) {
PyArg_Parse(PyList_GetItem(pKeys, i), "s", &cstr_key);
PyArg_Parse(PyList_GetItem(pValues, i), "s", &cstr_value);
//cout << cstr<< " "<< cstr2 <<endl;
my_map.emplace(cstr_key, cstr_value);
}
for (auto x : my_map)
{
cout << x.first << " : " << x.second << endl;
}
system("pause");
// Builds the name of a callable class
auto python_class = PyDict_GetItemString(pdict, "MyClass1");
system("pause");
if (python_class == nullptr) {
cout << "Fails to get the Python class.\n";
return 1;
}
//Py_DECREF(pdict);
cout << python_class;
PyObject* object;
// Creates an instance of the class
if (PyCallable_Check(python_class)) {
object = PyObject_CallObject(python_class, args);
Py_DECREF(python_class);
}
else {
std::cout << "Cannot instantiate the Python class" << endl;
Py_DECREF(python_class);
return 1;
}
auto val = PyObject_CallMethod(object, "is_jit_model_available", NULL);
if (!val)
cout << "error!";
cout << val;
Когда я пытаюсь запустить этот код, я получаю этот вывод, который показывает содержимое карт:
size: 5
__doc__ : Module1
__loader__ : Module1
__name__ : Module1
__package__ : Module1
__spec__ : Module1
Так что это результат PyModule_GetDict(module);
Однако, когда дело доходит до извлечения класса из этого модуля, он завершается неудачей, то есть PyDict_GetItemString(pdict, "MyClass1");
возвращает null
.
Мне кажется, что сам обработчик модуля не прав, и это, вероятно, потому, что, возможно, он не указывает на фактический модуль, что означает, что я полностью не смог импортировать и получить дескриптор этого модуля.
Поэтому я не могу придумать какой-либо другой способ, который позволяет мне импортировать модуль и использовать его следующим образом.
Что мне здесь не хватает?