解决方法
为了能够在Java中迭代Object,它需要实现Iterable.这又需要一个名为iterator()的成员函数,它返回一个合适的Iterator实现.
从你的问题来看,你不清楚你在地图中使用了什么类型,以及你是否希望能够迭代对(如在C中),键或值.三种变体的解决方案基本相似,下面我的例子选择了值.
首先,我用来测试这个SWIG接口文件的前导码:
%module test %include "std_string.i" %include "std_map.i"
为了实现可迭代的映射,我已经声明,定义并包装了SWIG接口文件中的另一个类.这个类,MapIterator为我们实现了Iterator接口.它是Java和包装C的混合体,其中一个比另一个更容易编写.首先是一些Java,一个类型映射,它给它实现的接口,然后是Iterable接口所需的三个方法中的两个,作为一个typemap给出:
%typemap(javainterfaces) MapIterator "java.util.Iterator<String>" %typemap(javacode) MapIterator %{ public void remove() throws UnsupportedOperationException { throw new UnsupportedOperationException(); } public String next() throws java.util.NoSuchElementException { if (!hasNext()) { throw new java.util.NoSuchElementException(); } return nextImpl(); } %}
然后我们提供MapIterator的C部分,它具有除了抛出next()部分的异常以及迭代器所需的状态(用std :: map自己的const_iterator表示)之外的所有私有实现.
%javamethodmodifiers MapIterator::nextImpl "private"; %inline %{ struct MapIterator { typedef std::map<int,std::string> map_t; MapIterator(const map_t& m) : it(m.begin()),map(m) {} bool hasNext() const { return it != map.end(); } const std::string& nextImpl() { const std::pair<int,std::string>& ret = *it++; return ret.second; } private: map_t::const_iterator it; const map_t& map; }; %}
最后我们需要告诉SWIG我们正在包装的std :: map实现了Iterable接口并提供了一个额外的成员函数,用于包装std :: map,它返回我们刚写的MapIterator类的新实例:
%typemap(javainterfaces) std::map<int,std::string> "Iterable<String>" %newobject std::map<int,std::string>::iterator() const; %extend std::map<int,std::string> { MapIterator *iterator() const { return new MapIterator(*$self); } } %template(MyMap) std::map<int,std::string>;
这可能是更通用的,例如用宏来隐藏地图的类型,这样如果你有多个地图,那么就像使用%模板一样“调用”适当地图的宏.
原始类型的映射也有一点点复杂 – 你需要安排Java端使用Double / Integer而不是double / int(我相信是自动装箱这个术语),除非你决定在这种情况下包装对你可以与原始成员配对.