虚函数表是每个对象一个还是每个类一个呢?

kesaihao862 2008-10-31 04:38:25
如题
...全文
2047 34 打赏 收藏 转发到动态 举报
写回复
用AI写文章
34 条回复
切换为时间正序
请发表友善的回复…
发表回复
zjljqlbtx 2008-11-09
  • 打赏
  • 举报
回复
每个对象一个
tienchiu 2008-11-09
  • 打赏
  • 举报
回复
[Quote=引用 25 楼 xmczr 的回复:]
类中不包含虚函数表,只包含一个指向该类虚函数表的指针,虚函数表是编译器保存在另外的地方,而不是类中.
[/Quote]

个人支持这种观点
lzh9955 2008-11-09
  • 打赏
  • 举报
回复
期待中!!!
chary8088 2008-11-07
  • 打赏
  • 举报
回复
构造函数不可以用虚函数,析构可以虚函数也就是这个原因
jackzhhuang 2008-11-05
  • 打赏
  • 举报
回复
谁想深入理解,看《深入理解C++对象模型》吧,三言两语,还真难说清。
xxgamexx 2008-11-05
  • 打赏
  • 举报
回复
谢谢LS的解答~
lann64 2008-11-04
  • 打赏
  • 举报
回复
[Quote=引用 27 楼 xxgamexx 的回复:]
我没理解:1。虚函数表指针确实是编译器加上去的,但是这个指针是由构造函数初始化,由析构函数破坏的。(可能又有人说构造函数初始化虚表指针的语句还是编译器加的,我无语)这句话~~
[/Quote]
他的意思是从代码上看,虚函数指针的初始化指令是附在构造函数后面,其他代码前面的。事实上现在编译器实现都是这么做的,从这个意义上看,说是构造函数初始化也说得通,不过虚函数指针还有重置的动作,这个可不是附在构造函数后面了,呵呵。
再说这个初始化指令放在哪纯属编译器实现上的事,标准上可根本没提及。再说一般人理解的构造函数功能里好像也不包含虚表指针初始化的功能,只把这个功能留给编译器去做了,至于编译器什么时候做,都不大关心的。
恐怕这个也是大家诟病C++的地方之一,c++有时候在背后偷偷做些事,不告诉程序员。(跟过程式语言比)
xxgamexx 2008-11-04
  • 打赏
  • 举报
回复
[Quote=引用 21 楼 wohow 的回复:]
按照C/C++的语法,指针和对象确实是截然不同的东西,但是我们分析问题时还要看到问题的实质。
当我们取得一个指向对象的指针后,我们可以通过这个指针访问这个对象,和取得这个对象本身具有完全相同的访问能力,因此拥有一个指向对象的指针可以看作就是拥有了这个对象。如果这样还有人说指针和对象就是不同的话,那么我只能说:阁下的治学态度真是很严谨。
另外18楼的说法不妥:
1。虚函数表指针确实是编译器加上去的,但是…
[/Quote]

我没理解:1。虚函数表指针确实是编译器加上去的,但是这个指针是由构造函数初始化,由析构函数破坏的。(可能又有人说构造函数初始化虚表指针的语句还是编译器加的,我无语)这句话~~
wudeshou82666 2008-11-04
  • 打赏
  • 举报
回复
讨论很深刻啊,呵呵,菜鸟就不发表看法了
xmczr 2008-11-04
  • 打赏
  • 举报
回复
类中不包含虚函数表,只包含一个指向该类虚函数表的指针,虚函数表是编译器保存在另外的地方,而不是类中.
wohow 2008-11-04
  • 打赏
  • 举报
回复
如果人家讨论问题都是基于语法角度而不看问题的实质,我就不用再和他讨论了
lann64 2008-11-04
  • 打赏
  • 举报
回复
[Quote=引用 21 楼 wohow 的回复:]
按照C/C++的语法,指针和对象确实是截然不同的东西,但是我们分析问题时还要看到问题的实质。
当我们取得一个指向对象的指针后,我们可以通过这个指针访问这个对象,和取得这个对象本身具有完全相同的访问能力,因此拥有一个指向对象的指针可以看作就是拥有了这个对象。如果这样还有人说指针和对象就是不同的话,那么我只能说:阁下的治学态度真是很严谨。
另外18楼的说法不妥:
1。虚函数表指针确实是编译器加上去的,但是…
[/Quote]
这个不能怪别人,是你自己太不严谨了。可访问和拥有是两个不同的概念,最简单的例子是友元。一般逻辑理解,可访问是单向的,拥有是双向的(A拥有B,B是A的一部分)。显然仅仅由类内一个指针指向的对象不能保证是类的成员,也不能保证这个对象可以访问类内的其他成分。这和类拥有的成员完全不能等同。
另外顺便说一句,虚函数表指针仅仅是编译时加上去的,构造函数不会初始化这个指针,也不能初始化这个指针。
wohow 2008-11-04
  • 打赏
  • 举报
回复
写错了,应该是在(void *)&*a处和(void *)&*a+sizeof(B)处
wohow 2008-11-04
  • 打赏
  • 举报
回复
按照C/C++的语法,指针和对象确实是截然不同的东西,但是我们分析问题时还要看到问题的实质。
当我们取得一个指向对象的指针后,我们可以通过这个指针访问这个对象,和取得这个对象本身具有完全相同的访问能力,因此拥有一个指向对象的指针可以看作就是拥有了这个对象。如果这样还有人说指针和对象就是不同的话,那么我只能说:阁下的治学态度真是很严谨。
另外18楼的说法不妥:
1。虚函数表指针确实是编译器加上去的,但是这个指针是由构造函数初始化,由析构函数破坏的。(可能又有人说构造函数初始化虚表指针的语句还是编译器加的,我无语)
2。每个对象只包含一个VPTR是不对的。例如
class A:public B, public C
其中B和C都有虚函数,那么A就有两个虚函数表,如果A *a = new A;那么两个虚函数表指针的地址分别在(void *)a处和(void *)a+sizeof(B)处
vcgaoshou 2008-11-03
  • 打赏
  • 举报
回复
每个类有一个或多个虚函数表
horris 2008-11-03
  • 打赏
  • 举报
回复
3,8,10楼正确!
wohow 2008-11-03
  • 打赏
  • 举报
回复
在C、C++里讨论指针时,指针本身只是一个整数,没有什么特别的含义,所以一般都把指针和指针指向的对象“等同”起来,比如
class A;
A *a = new A;
我们认为a就是类A的一个对象。如果按照10楼的说法,那么this指针只是一个指针,不代表对象喽?
pubb1986 2008-11-03
  • 打赏
  • 举报
回复
假如每个对象都有个虚函数表
那这些虚函数表不都是一样的吗?
那何必每个对象里都放一个虚函数表呢?

18楼是对的.

一个类对应一个虚函数表VTABLE,在每个带有VTABLE的类中,编译器秘密置一指针VPTR,指向这个虚函数表
每个对象只包含一个VPTR
e_sharp 2008-11-03
  • 打赏
  • 举报
回复
不考虑多重继承,一个类对应一个虚函数表VTABLE

chary8088 2008-11-03
  • 打赏
  • 举报
回复
一个类对应一个虚函数表VTABLE,在每个带有VTABLE的类中,编译器秘密置一指针VPTR,指向这个虚函数表
每个对象只包含一个VPTR
加载更多回复(14)
该课程由付强老师主讲,系统的、全面的、具体的讲解了java入门的知识。可以使初级的学员入门。Java入门Java的历史、Java的版本、Java的优势、软件行业前景Java开发环境搭建、编写Java入门练习虚拟机的运行机制、Java的平台无关性、虚拟机垃圾回收机制基础语法关键字、标识符、Java数据型分、基础数据型、取值范围变量、常量、三种注释、生成doc文档、运算符、表达式if语句、switch语句、嵌套判断for语句、while语句、do-while语句、嵌套循环对象面向过程和面向对象的区别的构成、访问修饰符、对象的内存分配this关键字、按值传递与按引用传递、对象的初始化顺序高级特性的继承、父子的初始化顺序、单继承性方法的覆盖、重载、super关键字、多态、instanceof关键字、强制型转换、static关键字、final关键字数组和枚举一维数组的应用及内存分配多维数组的应用及内存分配数组的复制、数组的按引用传递排序算法:冒泡、直接选择、插入选择、希尔、快速、归并、堆排序Arrays工具的应用枚举型的应用常见的使用Object的底层及应用、ObjectsString的底层及应用、正则表达式StringBuffer和StringBuilder的底层及应用Math的应用、包装的应用日期的应用:Date、DateFormat、Calendar、LocalDateTime、Instant、LocalDate、MonthDay、ZonedDateTime、YearMonth、新旧日期转换BigInteger和BingDecimal、DecimalFormatSystem、Scanner抽象和接口抽象的规则及应用接口的规则及应用默认方法、静态方法、函数式接口、Lambda表达式异常异常的定义异常的处理:抓(try-catch-finally)、抛(throws)异常的分、自定义异常的应用、throw关键字集合框架集合框架结构:接口、实现Collection接口的方法、Set接口的方法、List接口的方法、Map接口的方法Array、Linked、Hash、Tree底层实现原理泛型的作用、Collections工具、历史集合I/O流Path的原理及应用、Files的原理及应用文件字节流FileInputStream的原理及应用对象、缓冲流、数据流的原理及应用字符流的原理及应用多线程进程与线程的概念、查看线程对象Java内存模型线程的创建与启动:扩展Thread、实现Runnable接口、实现Callable接口、线程池线程状态的转换:新建、就绪、运行、阻塞、死亡线程的调度:sleep、yield、join、interrupt、wait后台线程、定时任务线程的并发与同步、同步锁、同步块、线程安全的Lock接口、CAS、volatile修饰符内部成员内部、本地内部的应用匿名内部的原理及引用、Lambda表达式设计模式基础设计模式概念、框架概念7大设计原则:开闭原则、依赖倒置原则、单一职责原则、接口隔离原则、迪米特原则、里氏替换原则、合成复用原则单例模式、工厂模式、模板模式、代理模式、装饰模式、适配器模式、外观模式、策略模式、观察者模式、命令模式、备忘录模式、观察者模式反射反射包Class的使用反射获取属性、方法、构造器通过反射创建对象/通过反射调用方法反射的应用

64,654

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧