例子1:属性的覆盖
#include "stdafx.h"

class A {
public:
int i;
A() { i=1; }
};

class B: public A {
public:
int i;
B() { i=2; }
};

class C: public B {
public:
int i;
C() { i=3; }
};

int main(int argc, char* argv[])
{
A* a1 = new A(); printf(" %d\n", a1->i);
A* a2 = new B(); printf(" %d\n", a2->i);
A* a3 = new C(); printf(" %d\n", a3->i);

printf(" %d\n", ((B*)a2)->i);
printf(" %d\n", ((B*)a3)->i);
printf(" %d\n", ((C*)a3)->i);

return 0;
}

输出:
1
1
1
2
2
3
结论:按照指针的定义类型来读取相关类的属性。大概三个属性在内存里都存在。有虚函数,但是没有虚数据成员!指针是什么类型(不管它指向什么对象),数据成员就是什么。

printf(" %d\n", sizeof(*a1)); // 这里,也是严格按照指针类型走。
printf(" %d\n", sizeof(*a2));
printf(" %d\n", sizeof(*a3));

同理:
int main(int argc, char* argv[])
{
A a1; printf(" %d\n", a1.i);
B a2; printf(" %d\n", a2.i);
C a3; printf(" %d\n", a3.i);

printf(" %d\n", ((A)a2).i); // 向上强制转换,变成A类型对象,则读取A的i
printf(" %d\n", ((B)a3).i); // 向上强制转换,变成B类型对象,则读取B的i

return 0;
}
输出:
1
2
3
1
2
结论:按照对象类型来读取相关类的数据成员。完全与Java一致。
(与Java比较,Java只有指针类型,而没有对象类型,即只要研究前一种情况即可)

例子2.1:误入函数参数的强制转换
#include "stdafx.h"

class A {
public:
virtual void get(int i) { printf("int in A: %d\n", i); }
virtual void get(double d) { printf("double in A: %f\n", d); }
};

class B: public A {
public:
virtual void get(int i) { printf("int in B: %d\n", i); }
};

int main(int argc, char* argv[])
{
A* pA = new A(); pA->get(1); pA->get(2.2);
A* pAB = new B(); pAB->get(1); pAB->get(2.2);
B* pB = new B(); pB->get(1); pB->get(2.2); // warning

return 0;
}
编译:
warning C4244: 'argument' : conversion from 'const double' to 'int', possible loss of data
输出:
int in A: 1
double in A: 2.200000
int in A: 1
double in A: 2.200000
int in B: 1
int in B: 2 // 注意,C++更倾向于做参数的强制转换,好继续调用int,而不是调用父类函数的get(double d)。但这不是覆盖父类同名函数覆盖

int main(int argc, char* argv[])
{
A a; a.get(1); a.get(2.2);
A ab; ab.get(1); ab.get(2.2);
B b; b.get(1); b.get(2.2); // warning

return 0;
}
编译:
warning C4244: 'argument' : conversion from 'const double' to 'int', possible loss of data
输出:
int in A: 1
double in A: 2.200000
int in A: 1
double in A: 2.200000
int in B: 1
int in B: 2 // 注意,C++更倾向于做参数的强制转换,好继续调用int,而不是调用父类函数的get(double d)。但这不是覆盖父类同名函数覆盖

结论:函数参数强制转换后的调用优先级比父类虚拟函数更高

注意:
1. C++函数不是默认virtual的。所以把上面的virtual都去掉的话,也会掉入那个参数强制转换的陷阱。
2. C++默认属性是private(如果不写明的话)

例子2.2:似乎存在子类一个函数覆盖父类全部同名函数的问题(还是有点怀疑),这点与Java不一样。
#include "stdafx.h"

class A {
private:
int i;
public:
A() { i=3; }
virtual void get(int i) { printf("int in A: %d\n", i); }
virtual void get(A a) { printf("double in A: %d\n", i); } // 注意,如果这里写%f编译没错,运行出错。可能double占的字节更多吧。
};

class B: public A {
private:
int i;
public:
B() { i=4; }
virtual void get(int i) { printf("int in B: %d\n", i); }
};

int main(int argc, char* argv[])
{
A a; a.get(1); a.get(a);
A ab; ab.get(1); ab.get(a);
B b; b.get(1);
b.get(a); // 编译不过,说不能把A转成int

return 0;
}

注意,这个例子还深挖一下,因为有一个private i,如果同名函数调用的话,到底调用谁呢?试了,数据成员跟着对象类型走。

例子3:测试静态函数覆盖问题
#include "stdafx.h"

class A {
private:
int i;
public:
A() { i=3; }
static void get(int i) { printf("int in A: %d\n", i); }
};

class B: public A {
private:
int i;
public:
B() { i=4; }
static void get(int i) { printf("int in B: %d\n", i); }
};

int main(int argc, char* argv[])
{
A::get(1);
B::get(2);
A a; a.get(1);
B b; b.get(2);
((A)b).get(3); // 里,按转换后的A类型调用静态函数
A* pa = new A(); pa->get(1);
A* pab = new B(); pab->get(1); // 这里,按指针类型A调用静态函数(静态函数是死脑筋,死跟着最简单的指针类型走,不像虚函数那样脑筋灵活)
B* pb = new B(); pb->get(2);
((A*)pb)->get(2); // 这里,按转换后的指针类型调用静态函数

return 0;
}
输出:
int in A: 1
int in B: 2
int in A: 1
int in B: 2
int in A: 3
int in A: 1
int in A: 1
int in B: 2
int in A: 2

结论:静态函数跟着指针类型走(毕竟静态函数不是虚拟函数),而不是指针指向的对象类型走。这一点,跟数据成员一致。

注意:静态函数不能调用对象的属性,以下是无法编译通过的:

class A {
private:
int i;
public:
A() { i=3; }
static void get() { printf("int in A: %d\n", i); }
};

C++的同名属性(没有虚拟属性)、同名普通函数、同名静态函数(没有虚拟静态函数),是否被覆盖的更多相关文章

  1. schema中的虚拟属性方法

    schema中的虚拟属性方法相当于vue中的计算属性,它是通过已定义的schema属性的计算\组合\拼接得到的新的值 var personSchema = new Schema({ name: { f ...

  2. C#中对虚拟属性和抽象属性的重写有什么异同

           public abstract class A         {             //抽象属性不能有实现代码             public abstract strin ...

  3. Python类属性,实例属性

    1.Python类数据属性:定义在类里面但在函数外面的变量,它们都是静态的. #一段很简单的代码,但反应了很多 >>> class A(): a=1 #一个类里面有个属性a > ...

  4. Python中的类属性、实例属性与类方法、静态方法

    1.什么是类对象,实例对象 类对象:类名 实例对象:类创建的对象 2.类属性就是类对象所拥有的属性,它被所有类对象的实例对象所共有,在内存中只存在一个副本,这个和C++.Java中类的静态成员变量有点 ...

  5. JS 本地属性与继承属性

    判断是否拥有某种属性 1.in 运算符 var obj = {name:'jack'}; alert('name' in obj); // --> true alert('toString' i ...

  6. C语言中的作用域、链接属性与存储属性

    C语言中的作用域.链接属性与存储属性 一.作用域(scope) 代码块作用域 表示{}之间的区域,下例所示,a可以在不同的代码块里面定义. #include<stdio.h> int ma ...

  7. Python3 之 类属性与实例属性

    1.类属性与实例属性 类属性就相当与全局变量,实例对象共有的属性,实例对象的属性为实例对象自己私有. 类属性就是类对象(Tool)所拥有的属性,它被所有类对象的实例对象(实例方法)所共有,在内存中只存 ...

  8. python的类属性、实例属性、类方法、静态方法

    类属性 就像如下代码: class Person: name = "张三" # 共有类属性 __age = 18 # 私有类属性 在类中直接定义的属性就是类属性,它被所有的实例对象 ...

  9. Python基础:17类和实例之一(类属性和实例属性)

    1:类通常在一个模块的顶层进行定义.对于Python来说,声明与定义类是同时进行的. 2:类属性仅与其类相绑定,类数据属性仅当需要有更加“静态”数据类型时才变得有用,这种属性是静态变量.它们表示这些数 ...

  10. The related functions and attributes for managing attributes - 操作属性的重要属性和函数

    特性 property 都是类属性(静态变量),但是特性管理的其实是实例属性的存取, ****** 回顾 -'类方法' classmethod 和 '静态方法' staticmethod 皆可以访问类 ...

随机推荐

  1. cast() 函数进行类型转换

    service_fee  字段定义:`service_fee`  decimal(14,4) NULL DEFAULT NULL COMMENT '手续费金额,含税' , 需求:service_fee ...

  2. CodeForces 1100F Ivan and Burgers

    CodeForces题面 Time limit 3000 ms Memory limit 262144 kB Source Codeforces Round #532 (Div. 2) Tags da ...

  3. 使用ros_driver运行velodyne16线激光雷达

    一.使用ros_driver运行VLP16 推荐网址: http://blog.csdn.net/littlethunder/article/details/51920681 https://www. ...

  4. 损坏的RAID5

    损坏的RAID5 string讀入卡cin 関同步 ios::sync_with_stdio(false) 由塊號映射到具體位置 塊號id對應第col個字符串 字符串開始的位置st #include& ...

  5. 接口自动化request库入门

    requests库7个主要方法 r= requsts.get(),主要属性: r.raise_for_status()方法内部判断r.status_code是否等于200不需要增加额外的if语句,该语 ...

  6. C# walls

    在学习C#的阶段中,我们一点一点的往前爬, 此代码需要添加selenium ,和 获取 引用. using Ivony.Html.Parser; using Ivony.Html; using Ope ...

  7. ubuntu16虚拟机迁移/移动/复制后无法上网

    修改grub配置 如果没有网卡,需要配置 sudo vi /etc/default/grub 将 GRUB_CMDLINE_LINUX="" 修改为 GRUB_CMDLINE_LI ...

  8. NDK下编译JNI

    NDK环境下编译JNI 下载demo.tar.gz然后解压 弄个套路 1.编辑build.sh设置好NDK目录 2.把cpp文件放到code下面 运行sh build.sh即可

  9. 用Jquery选择器计算table中的某一列某一行的合计

    核心算法: $('#tableId tr').each(function() { $(this).find('td:eq(columnIndex)').each(function() { totalA ...

  10. 【奇技淫巧】使用 SSH 转发 Sock5 流量

    标题:使用 SSH 转发 Sock5 流量 日期:2018-06-27 介绍:使用 ssh 来做个 sock5 的代理,穿透到内网中做后渗透 0x01. 基本信息 在 ubuntu(10.211.55 ...