c++ 对象模型 20

本文深入探讨C++中的虚函数与多态性,包括vcall机制、静态与动态绑定的区别,以及如何通过虚函数实现多态。通过实例说明了虚函数地址的查找过程,静态与动态类型的特性,以及虚函数如何根据对象的动态类型进行绑定。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

虚函数地址问题的vcall

class Myclass{
public:
    virtual void myfunc(){
        
    }
    virtual void myfunc2(){
        
    }
}

int main(){
    printf("Myclass:myfunc() = %p",&Myclass::myfunc);  // x86 4字节
    printf("Myclass:myfunc2() = %p",&Myclass::myfunc2);  // x86 4字节
    // 这里是实际的函数,但是函数调用时通过vcall中转了一下
    //  所以这里打印的是vcall函数的地址
    // vcall thunk  调整this指针用的偏移
    //               跳转到真正的虚函数中去
    
    
    Myclass *myclass = new Myclass();  // 这里存放的地址和上述地址不同
    
}

-------------------------------------------------------------------

静态类型与动态类型

class Base{
public:
    
}

class Derive :public Base{
public:
}

class Derive2: public Base{
public:
}

/*
静态类型: 对象定义时的类型 ,编译期间就能确定好
Base base;   base 的静态类型时Base
Derive derive;  derive 的静态类型时 Derive
Base *pbase;  pbase 的静态类型时Base *
Base *pbase2 = new Derive();  pbase2 的静态类型时Base *


动态类型: 对象目前所指向的类型  运行时所指向的类型  一般只有指针和引用才有动态类型的说法
        一般就都是指父类的指针或者引用,  另外动态类型在执行过程中可以改变
    Base *pbase2 = new Derive();   动态类型时Derive  
    Base *pbase;     至少目前没有动态类型  他没有指向任何对象
    pbase = new Derive2()    这时候动态类型就是Derive2
    
    
*/

-------------------------------------------------------------------

静态绑定 : 绑定的时静态类型, 所对应的函数或者属性依赖与对象的静态类型   发生在编译期
动态绑定:  绑定的时多年柜台类型 所对应函数或者属性依赖于对象的动态绑定  发生在运行期

普通成员函数时静态绑定, 而虚函数时动态绑定
缺省参数一般是静态绑定

class Base{
public:
    void myfunc(){
        cout << "Base::myfunc()" <<endl;
    }
}

class Derive :public Base{
public:
    void myfunc(){
        cout << "Derive::myfunc()" << endl;
    }
}

class Derive2: public Base{
public:
}

int main (){
    Derive derive;
    Derive *pderive = &derive;
    pderive->myfunc();         // Derive::myfunc()
    
    Base *pbase = &derive;
    pbase->myfunc();           // Base::myfunc()
    // 这时候就是体现出普通成员函数时静态绑定  ,
    // 这里到底调用时父类的myfunc 还是子类的myfunc 取决于调用者自身的静态类型
    // 所以里调用Base的myfunc
    // 结论 :  不应该在子类中重新定义一个继承来的非虚函数
}


class Base{
public:
    void myfunc(){
        cout << "Base::myfunc()" <<endl;
    }
    virtual void myvirfunc(){
        cout << "Base::myvirfunc()" <<endl;
    }
}

class Derive :public Base{
public:
    void myfunc(){
        cout << "Derive::myfunc()" << endl;
    }
    virtual void myvirfunc(){
        cout << "Derive::myvirfunc()" <<endl;
    }
}

class Derive2: public Base{
public:
}

int main (){
    Derive derive;
    Derive *pderive = &derive;

    
    Base *pbase = &derive;
    pbase->myfunc(); 
    
    Base base;
    pderive->myvirfunc();
    pbase->myvirfunc();
    pbase = &base;
    pbase->myvirfunc();
    // 虚函数是动态绑定 : pderive 绑定的时Derive所以调用Derive的myvirfunc()  
    // 同样    pbase  绑定的Base *pbase = &derive;  所以也是调用Derive的myvirfunc()
    // 最后   pbase = &base;  调用的时Base的myvirfunc();
    
    // 总结 myvirfunc() 是虚函数 这里到底执行哪个myvirfunc()取决于调用者的动态类型
    

class Base{
public:
    void myfunc(){
        cout << "Base::myfunc()" <<endl;
    }
    virtual void myvirfunc(int value = 1){
        cout << "Base::myvirfunc(), value = "<< value <<endl;
    }
}

class Derive :public Base{
public:
    void myfunc(){
        cout << "Derive::myfunc()" << endl;
    }
    virtual void myvirfunc(int value = 2){
        cout << "Derive::myvirfunc(), value =" << value <<endl;
    }
}

int main (){
    Derive derive;
    Derive *pderive = &derive;

    
    Base *pbase = &derive;
    pbase->myfunc(); 
    
    Base base;
    pderive->myvirfunc();
    pbase->myvirfunc();
    pbase = &base;
    pbase->myvirfunc();
    /*
    Derive::myvirfunc() ,value = 2
    Derive::myvirfunc() ,value = 1   //  执行了子类的虚函数值  value 却是1
    Base::myvirfunc() ,value = 1
    */
    //这时候就体现了  缺省参数 的静态绑定

    


// 总结: 多态性

多态性这个概念 分两个方面 但是有一点, 多态 必须存在虚函数 没有虚函数 就不可能存在多态 并且 调用虚函数

    1.从代码实现上
        当我们掉一个虚函数时, 走的是不是通过查询虚函数表来找到虚函数入口地址 然后去执行虚函数
        那就是多态  如果不走这个途径 那就不是多态

class A{
public:
    virtual void virfunc(){
    }
}

int main(){
    A *a = new A();
    a->virfunc();  // 这就是多态
    
    A a1;
    a1.virfunc();   // 这不是多态
    
    A *pa = &a;
    pa->virfunc();   // 这个也是多态
    
}

    2. 从表现形式上来说
        有继承关系 ,有父类有子类 父类中必须有虚函数 (这意味着子类中一定有虚函数  
        派生类重写基类的虚函数
        父类指针或者指向引用子类对象
        当以父类指针或这引用调用子类中重写了的虚函数时候 我们就看出来多态 
        因为调用子类的虚函数

1. 用户与权限管理模块 角色管理: 学生:查看实验室信息、预约设备、提交耗材申请、参与安全考核 教师:管理课题组预约、审批学生耗材申请、查看本课题组使用记录 管理员:设备全生命周期管理、审核预约、耗材采购与分发、安全检查 用户操作: 登录认证:统一身份认证(对接学号 / 工号系统,模拟实现),支持密码重置 信息管理:学生 / 教师维护个人信息(联系方式、所属院系),管理员管理所有用户 权限控制:不同角色仅可见对应功能(如学生不可删除设备信息) 2. 实验室与设备管理模块 实验室信息管理: 基础信息:实验室编号、名称、位置、容纳人数、开放时间、负责人 功能分类:按学科(计算机实验室 / 电子实验室 / 化学实验室)标记,关联可开展实验类型 状态展示:实时显示当前使用人数、设备运行状态(正常 / 故障) 设备管理: 设备档案:名称、型号、规格、购置日期、单价、生产厂家、存放位置、责任人 全生命周期管理: 入库登记:管理员录入新设备信息,生成唯一资产编号 维护记录:记录维修、校准、保养信息(时间、内容、执行人) 报废处理:登记报废原因、时间,更新设备状态为 "已报废" 设备查询:支持按名称、型号、状态多条件检索,显示设备当前可用情况 3. 预约与使用模块 预约管理: 预约规则:学生可预约未来 7 天内的设备 / 实验室,单次最长 4 小时(可设置) 预约流程:选择实验室→选择设备→选择时间段→提交申请(需填写实验目的) 审核机制:普通实验自动通过,高危实验(如化学实验)需教师审核 使用记录: 签到 / 签退:到达实验室后扫码签到,离开时签退,系统自动记录实际使用时长 使用登记:填写实验内容、设备运行情况(正常 / 异常),异常情况需详细描述 违规管理:迟到 15 分钟自动取消预约,多次违规限制预约权限 4. 耗材与安全管理模块 耗材管理: 耗材档案:名称、规格、数量、存放位置、
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值