Хорошо, моя схема именования немного глупа, оставляя вам возможность найти что-то лучшее, но вы можете попробовать это:
// initial base class we can return from A without having to worry about templates
class Getter
{
public:
virtual ~Getter() { }
};
// intermediate class intended to be dynamic_casted to, see below
template<typename R>
class TheGetter : public Getter
{
public:
virtual R operator()() = 0;
};
// final implementation
template<typename T, typename R>
class TheTheGetter : public TheGetter<R>
{
T* t;
R (T::*getter)();
public:
TheTheGetter(T* t, R (T::*getter)())
: t(t), getter(getter)
{ }
R operator()() override
{
return (t->*getter)();
}
};
struct A
{
virtual ~A() { };
virtual std::string get_name() = 0;
// as virtual, cannot be a template, thus Getter cannot be either...
virtual Getter& get_getter(std::string name) = 0;
};
// let's make a template function of:
template<typename T>
T PropertyManager::get_property(string_type object_name, string_type property_name)
{
for(auto& o : objects)
{
if (o->get_name() == object_name)
{
auto& getter = o->get_getter(property_name);
// now with the intermediate class, we do not have to care for
// o's true type...
auto* theGetter = dynamic_cast<TheGetter<T>*>(&getter);
if(theGetter)
{
return (*theGetter)();
}
throw std::runtime_error("... is of bad type ...");
}
}
throw std::runtime_error("...");
}
class Test : public A
{
public:
Test()
: gInt(this, &Test::getInt),
gDouble(this, &Test::getDouble)
{ }
int getInt() { return 7; }
double getDouble() { return 10.12; }
std::string get_name() override
{
return "Test";
}
Getter& get_getter(std::string name) override
{
if(name == "int")
{
return gInt;
}
if(name == "double")
{
return gDouble;
}
throw std::runtime_error("...");
}
private:
TheTheGetter<Test, int> gInt;
TheTheGetter<Test, double> gDouble;
};
int main(int argc, char* argv[])
{
PropertyManager p;
p.add_a(std::make_shared<Test>());
auto vi = p.get_property<int>("Test", "int");
auto vd = p.get_property<double>("Test", "double");
std::cout << vi << ' ' << vd << std::endl;
return 0;
}