我已經將std::vector<std::vector<std::string > >
的一個最小包裝的例子放在一起,它不包含任何額外的SWIG文件(例如std_vector.i和std_string.i)。
我也把一個小的頭文件來測試我的實現有:
#include <vector>
#include <string>
#include <algorithm>
#include <iterator>
#include <iostream>
inline void print_vec(const std::vector<std::string>& v) {
std::copy(v.begin(),v.end(), std::ostream_iterator<std::string>(std::cout, "\n"));
}
inline void print_vec_vec(const std::vector<std::vector<std::string> >& v) {
std::for_each(v.begin(),v.end(),print_vec);
}
std::vector<std::vector<std::string> > make() {
static std::vector<std::string> test1;
static std::vector<std::string> test2;
static std::vector<std::vector<std::string> > ret;
test1.push_back("hello");
test2.push_back("world");
test2.push_back("another");
ret.push_back(test1);
ret.push_back(test2);
return ret;
}
這是我能想到的,最小的實施有效地行使生成的接口。
我寫的SWIG接口提供了一個std::vector
的骨架定義 - 這足以說服SWIG實際包裝這個東西。我們還擴展了它,針對我們關心的兩種情況,提供了__getitem__
的實現,這是您希望能夠使用的obj[x][y]
語法的最低要求。
%module Test
%{
#include "test.hh"
%}
namespace std {
template <typename T>
class vector {
};
}
%extend std::vector<std::vector<std::string> > {
std::vector<std::string> __getitem__(unsigned i) throw(std::out_of_range) {
return $self->at(i);
}
}
%extend std::vector<std::string> {
const char * __getitem__(unsigned i) throw(std::out_of_range) {
return $self->at(i).c_str();
}
}
%template (VecString) std::vector<std::string>;
%template (VecVecString) std::vector<std::vector<std::string> >;
%include "test.hh"
有一個竅門那裏c_str()
以避免包括std_string.i。該接口允許我做這樣的事情在Python:
Python 2.7.1+ (r271:86832, Apr 11 2011, 18:05:24)
[GCC 4.5.2] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> import Test
>>> t=Test.make()
>>> print t[0][0]
hello
>>>
它目前不會提高正確類型的Python異常的__getitem__
。你可以用%include "exception.i"
或%exception
這樣做,並在$action
附近編寫自己的try
/catch
。
您可能還想提供一個類似的實現__setitem__
以使其有用。
這可能不會比std_vector.i更快,或者直接轉換爲Python列表的home brew類型映射。總的來說,雖然我認爲不這樣做是一個好主意 - 使用現有的std_vector.i實現而不是重新創建輪子似乎更符合邏輯。
它是矢量<矢量>。我寫的問題,但由於一些未知錯誤,它顯示矢量>代替矢量<矢量>。 –
Saurabh
我這裏不明白的問題 - 你想換一個'的std :: VECTOR'不使用已經提供的接口。你也可以說你已經編寫了代碼'vector>'列表。你能用一個最簡單的例子來解釋你想要包裝什麼,以及爲什麼這些都不明智? –
Flexo
好吧,我不使用已經提供的接口,即std_vector.i,因爲它不是在我的組織中可用的(我們使用我們自己的編譯工具和庫,因此我們不能使用std_vector.i)。目前,我做這樣的事情: %採用矢量<矢量> * OUTPUT {矢量<矢量> *結果}; 類型映射(在,numinputs = 0)矢量<矢量> * OUTPUT(矢量<矢量> T){ 1 $ = T; ($ argout,fragment =「t_output_helper」)vector > * OUTPUT {result} $ result = t_output_helper($ result,my_function_to_convert_vectorToList($ 1)); } –
Saurabh