EDIT: I have a working solution now, https://wandbox.org/permlink/joS14MzfmmayNRd3, just want to know if it can be improved.
I have a map map<int, unique_ptr<Base>>
, whose values can also be of type unique_ptr<Derived>
. I have wrapper classes A and B which store Base and Derived in the map respectively (all Base or all Derived instances).
I want to introduce map semantics (reference binding) for A and B that works as follows:
// a is of type A, all values in map are of type Base
for (auto& [k, v] : a) {
// v should be of type Base&
}
// b is of type B, all values in map are type Derived
for (auto& [k, v] : b) {
// v should be of type Derived&
// can call functions that are in Derived but not in Base
}
I found boost's transform_iterator and iterator_adaptor that could potentially be useful (even better if there is a way to do it without boost), but I don't seem to be using them correctly. What is the right way to do this?
#include <memory>
#include <boost/iterator/transform_iterator.hpp>
#include <vector>
#include <unordered_map>
#include <iostream>
using namespace std;
class Base {
public:
Base(int k) : i(k) {}
virtual ~Base() {}
virtual int getV() {
return i;
}
int i = 0;
};
class Derived : public Base {
public:
Derived(int k) : Base(k) {}
int getV() override {
return j - i;
}
void setV(int p) {
i = p;
}
int j = 0;
};
typedef unordered_map<int, unique_ptr<Base>> BaseMap;
class A {
public:
A(vector<int> keys) {
for (auto& k : keys) {
m.emplace(k, make_unique<Base>(k));
}
}
class Converter {
public:
explicit Converter() {}
pair<BaseMap::key_type, reference_wrapper<Base>> operator()(BaseMap::value_type& p) const {
return {p.first, *p.second};
}
};
using MyIterator = boost::transform_iterator<Converter, typename BaseMap::iterator>;
MyIterator begin() {
return MyIterator(m.begin());
}
MyIterator end() {
return MyIterator(m.end());
}
protected:
BaseMap m;
};
class B : public A {
public:
B(vector<int> keys) : A(keys) {
m.clear(); // demo only, since we have to init A
for (auto& k : keys) {
m.emplace(k, make_unique<Derived>(k));
}
}
class Converter {
public:
explicit Converter() {}
pair<BaseMap::key_type, reference_wrapper<Derived>> operator()(BaseMap::value_type& p) const {
return {p.first, dynamic_cast<Derived&>(*p.second)};
}
};
using MyIterator = boost::transform_iterator<Converter, typename BaseMap::iterator>;
MyIterator begin() {
return MyIterator(m.begin());
}
MyIterator end() {
return MyIterator(m.end());
}
};
int main ()
{
A a({1,2,3,4});
B b({1,2,3,4});
for (auto [k, v] : a) {
cout << v.get().getV() << " ";
}
cout << endl;
for (auto [k, v] : b) {
cout << v.get().getV() << " ";
v.get().setV(42);
}
cout << endl << "after setV:\n";
for (auto [k, v] : b) {
cout << v.get().getV() << " ";
}
return 0;
}