доверенность - PullRequest
       32

доверенность

2 голосов
/ 08 января 2020

Вызов приведенного ниже кода из python:

from f_p import form_p
print(form_p([1,2,3]))

Дает мне ошибку ниже

Ошибка:

(<f_p.mapiv; proxy of <Swig Object of type 'std::map< char,int > *' at 0x7f09778fc270> >,)

Как решить это? Я пытаюсь создать оболочку для моего cpp кода в Swig. Файлы кодов:

f_p.i:

%module f_p
#define SWIGPYTHON_BUILTIN

%{
  #include "numpy/arrayobject.h"
  #define SWIG_FILE_WITH_INIT  /* To import_array() below */
  #include "f_p.h"
%}

%include "std_map.i"
%import "std_deque.i" 
// %include "numpy.i"
%import "std_vector.i" 

#include <deque>

%template() std::vector<int>;

%template (map) std::map<char,int>;
%template(MapDeque) std::deque<std::map<char, int>>;


%include "f_p.h"

f_p. cpp:

#include <deque> 
#include <iostream> 
using namespace std; 

#include <vector>
#include <map>  

deque<map<char, int>> form_p(vector<int> inp_list) 
{
    map<char, int> my_map = {
    { 'A', 1 },
    { 'B', 2 },
    { 'C', 3 }
    };
    deque<map<char, int>> mydeque;
    mydeque.push_back(my_map); 
    return mydeque;
}

f_p.h:

#ifndef F_P_H
#define f_P_H
#include <stdio.h>
#include <deque>
#include <map>
#include <vector>

/* Define function prototype */
std::deque<std::map<char, int>> form_p(std::vector<int> inp_list) ;
#endif

build. sh:

rm *.o f_p_wrap.cpp _f_p.so f_p.py
rm -rf __pycache__

g++ -O3 -march=native -fPIC -c f_p.cpp

swig -python -c++ -o f_p_wrap.cpp f_p.i

# Next, compile the wrapper code:

g++ -O3 -march=native -w -fPIC -c $(pkg-config --cflags --libs python3) -I /home/kriti/anaconda3/lib/python3.7/site-packages/numpy/core/include f_p.cpp f_p_wrap.cpp

g++ -std=c++11 -O3 -march=native -shared f_p.o f_p_wrap.o -o _f_p.so -lm

Я не могу чтобы получить выход. Я не знаю, как работать с вещами deque. для карты и вектора я готов выдавать результаты, но не с deque, а когда на карте есть char.

1 Ответ

2 голосов
/ 08 января 2020

То, что вы пометили как «Ошибка», является правильным (или, по крайней мере, ожидаемым) результатом: deque транслируется в кортеж Python. Чтобы добраться до вашей карты, просто получите доступ к первому элементу, а затем к карте:

>>> from f_p import form_p
>>> print(form_p([1,2,3]))
(<f_p.map; proxy of <Swig Object of type 'std::map< char,int > *' at 0x7f16259a1c60> >,)
>>> d = form_p([1,2,3])
>>> len(d)
1
>>> d[0]
<f_p.map; proxy of <Swig Object of type 'std::map< char,int > *' at 0x7f16259aeb40> >
>>> m = d[0]
>>> len(m)
3
>>> m.keys()
['A', 'B', 'C']
>>> m['B']
2
>>> for m in d:
...     print m.keys()
... 
['A', 'B', 'C']
>>> 

РЕДАКТИРОВАТЬ : продолжение на основе комментариев. Для std::deque уже существует типовая карта «out», которая, к сожалению, более точно соответствует, поэтому, если вы не укажете std::deque<std::map<char, int>> form_p полностью (т.е. с именем функции), она не будет соответствовать, поэтому в следующем примере вместо нее используется «ret» , чтобы применить ко всем вашим функциям, которые возвращают такие запросы. Добавить к f_p.i:

%typemap(ret) std::deque<std::map<char, int>> {
  $result = PyTuple_New($1.size());
  for (int i = 0; i < (int)$1.size(); ++i)
    PyTuple_SetItem($result, i, swig::traits_from<std::map<char, int>>::asdict($1[i]));
}

Этот код создает кортеж (используйте PyList_New и PyList_SetItem, если вы предпочитаете список python), затем перебирает записи в deque и конвертирует их до python диктов. Вызов asdict - это сгенерированная python функция, которую вы также можете использовать в python коде постобработки в .i, если хотите,

С этим на месте результат:

>>> from f_p import form_p
>>> print(form_p([1,2,3]))
({'A': 1, 'C': 3, 'B': 2},)
>>> 
...