假设我的课程Foo
和Bar
设置如下:
class Foo
{
public:
int x;
virtual void printStuff()
{
std::cout << x << std::endl;
}
};
class Bar : public Foo
{
public:
int y;
void printStuff()
{
// I would like to call Foo.printStuff() here...
std::cout << y << std::endl;
}
};
在代码中注释,我希望能够调用我所覆盖的基类函数。在Java中,有super.funcname()
语法。这在C ++中是否可行?
答案 0 :(得分:402)
C ++语法是这样的:
class Bar : public Foo {
// ...
void printStuff() {
Foo::printStuff(); // calls base class' function
}
};
答案 1 :(得分:109)
是,
class Bar : public Foo
{
...
void printStuff()
{
Foo::printStuff();
}
};
它与Java中的super
相同,除了它允许在有多个继承时从不同的基础调用实现。
class Foo {
public:
virtual void foo() {
...
}
};
class Baz {
public:
virtual void foo() {
...
}
};
class Bar : public Foo, public Baz {
public:
virtual void foo() {
// Choose one, or even call both if you need to.
Foo::foo();
Baz::foo();
}
};
答案 2 :(得分:64)
有时你需要调用基类的实现,当你不在派生函数中时......它仍然可以工作:
struct Base
{
virtual int Foo()
{
return -1;
}
};
struct Derived : public Base
{
virtual int Foo()
{
return -2;
}
};
int main(int argc, char* argv[])
{
Base *x = new Derived;
ASSERT(-2 == x->Foo());
//syntax is trippy but it works
ASSERT(-1 == x->Base::Foo());
return 0;
}
答案 3 :(得分:27)
以防万一你为班上的很多功能做这件事:
class Foo {
public:
virtual void f1() {
// ...
}
virtual void f2() {
// ...
}
//...
};
class Bar : public Foo {
private:
typedef Foo super;
public:
void f1() {
super::f1();
}
};
如果你想重命名Foo,这可能会节省一些写作。
答案 4 :(得分:6)
如果要从派生类中调用基类函数,可以简单地在覆盖函数内调用提及基类名称(如 Foo :: printStuff())。
代码在这里
#include <iostream>
using namespace std;
class Foo
{
public:
int x;
virtual void printStuff()
{
cout<<"Base Foo printStuff called"<<endl;
}
};
class Bar : public Foo
{
public:
int y;
void printStuff()
{
cout<<"derived Bar printStuff called"<<endl;
Foo::printStuff();/////also called the base class method
}
};
int main()
{
Bar *b=new Bar;
b->printStuff();
}
同样,您可以在运行时确定使用该类的对象(派生或基础)调用哪个函数。但是这需要您在基类中的函数必须标记为虚拟。
下面的代码
#include <iostream>
using namespace std;
class Foo
{
public:
int x;
virtual void printStuff()
{
cout<<"Base Foo printStuff called"<<endl;
}
};
class Bar : public Foo
{
public:
int y;
void printStuff()
{
cout<<"derived Bar printStuff called"<<endl;
}
};
int main()
{
Foo *foo=new Foo;
foo->printStuff();/////this call the base function
foo=new Bar;
foo->printStuff();
}
答案 5 :(得分:1)
如果有多个层次的继承,你可以指定直接基类,即使实际实现在较低的层次。
class Foo
{
public:
virtual void DoStuff ()
{
}
};
class Bar : public Foo
{
};
class Baz : public Bar
{
public:
void DoStuff ()
{
Bar::DoStuff() ;
}
};
在本例中,Baz 类指定了 Bar::DoStuff(),尽管 Bar 类不包含 >做事。这是一个细节,Baz 不需要知道。
调用 Bar::DoStuff 显然比调用 Foo::DoStuff 更好,以防 Bar 的更新版本也覆盖此方法。
答案 6 :(得分:0)
检查一下......
#include <stdio.h>
class Base {
public:
virtual void gogo(int a) { printf(" Base :: gogo (int) \n"); };
virtual void gogo1(int a) { printf(" Base :: gogo1 (int) \n"); };
void gogo2(int a) { printf(" Base :: gogo2 (int) \n"); };
void gogo3(int a) { printf(" Base :: gogo3 (int) \n"); };
};
class Derived : protected Base {
public:
virtual void gogo(int a) { printf(" Derived :: gogo (int) \n"); };
void gogo1(int a) { printf(" Derived :: gogo1 (int) \n"); };
virtual void gogo2(int a) { printf(" Derived :: gogo2 (int) \n"); };
void gogo3(int a) { printf(" Derived :: gogo3 (int) \n"); };
};
int main() {
std::cout << "Derived" << std::endl;
auto obj = new Derived ;
obj->gogo(7);
obj->gogo1(7);
obj->gogo2(7);
obj->gogo3(7);
std::cout << "Base" << std::endl;
auto base = (Base*)obj;
base->gogo(7);
base->gogo1(7);
base->gogo2(7);
base->gogo3(7);
std::string s;
std::cout << "press any key to exit" << std::endl;
std::cin >> s;
return 0;
}
输出
Derived
Derived :: gogo (int)
Derived :: gogo1 (int)
Derived :: gogo2 (int)
Derived :: gogo3 (int)
Base
Derived :: gogo (int)
Derived :: gogo1 (int)
Base :: gogo2 (int)
Base :: gogo3 (int)
press any key to exit
最好的方法是使用 base :: function ,如@sth
答案 7 :(得分:0)
是的,你可以打电话给它。用于在子类中调用父类函数的C ++语法是
class child: public parent {
// ...
void methodName() {
parent::methodName(); // calls Parent class' function
}
};
阅读有关函数overriding的更多信息。