我想将访问者模式与智能指针结合使用,但似乎编译器无法检测子类并匹配相应的函数.
我写了一个显示问题的SSCE:
test.hpp:
class Visitor;
class Area {
public:
virtual void visit(const Visitor& visitor) = 0;
};
class Area1 : public Area {
public:
virtual void visit(const Visitor& visitor);
};
class Area2 : public Area {
public:
virtual void visit(const Visitor& visitor);
};
class Visitor {
public:
virtual void visit(const Area1 &area) const = 0;
virtual void visit(const Area2 &area) const = 0;
};
Run Code Online (Sandbox Code Playgroud)
TEST.CPP:
#include "test.hpp"
void Area1::visit(const Visitor& visitor) {
visitor.visit(*this);
}
void Area2::visit(const Visitor& visitor) {
visitor.visit(*this);
}
Run Code Online (Sandbox Code Playgroud)
main.cpp中
#include <iostream>
#include "boost/shared_ptr.hpp"
#include "test.hpp"
class FooVisitor : public Visitor {
virtual void visit(const Area1 &area) const {
std::cout <<"visit area1" <<std::endl;
};
virtual void visit(const Area2 &area) const {
std::cout <<"visit area2" <<std::endl;
}
};
int main(int argc, char** argv) {
boost::shared_ptr<Area> p(new Area2);
FooVisitor visitor;
visitor.visit(*p);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
该生产线visitor.visit(*P)给出了这样的错误:
main.cpp: In function ‘int main(int, char**)’:
main.cpp:17:21: error: no matching function for call to ‘FooVisitor::visit(Area&)’
main.cpp:17:21: note: candidates are:
main.cpp:6:18: note: virtual void FooVisitor::visit(const Area1&) const
main.cpp:6:18: note: no known conversion for argument 1 from ‘Area’ to ‘const Area1&’
main.cpp:9:18: note: virtual void FooVisitor::visit(const Area2&) const
main.cpp:9:18: note: no known conversion for argument 1 from ‘Area’ to ‘const Area2&’
Run Code Online (Sandbox Code Playgroud)
我如何帮助编译器实现访问(const Area2&)而不需要尝试使用dynamic_cast到每个子类型?
这是行不通的,因为它的类型*p确实是Area&(基于指针的类型).但是,你错了.要正确应用访问者模式,您应该调用已visit()定义的内容Area,以获得正确的调度:
int main(int argc, char** argv) {
boost::shared_ptr<Area> p(new Area2);
FooVisitor visitor;
p->visit(visitor);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
这Area::visit()首先是该功能的目的.