C++的动态联编与静态联编【转载】

C++的动态联编与静态联编【转载】

【摘要】:本文阐述了静态联编和动态联编的概念和区别,通过具体实例分析了实现动态联编的条件,指出了虚函数是实现动态联编的基础。

【关键词】:静态联编;动态联编;虚函数

在C++中,联编是指一个计算机程序的不同部分彼此关联的过程。按照联编所进行的阶段不同,可分为两种不同的联编方法:静态联编和动态联编。

    1. 静态联编

 

  静态联编是指联编工作在编译阶段完成的,这种联编过程是在程序运行之前完成的,又称为早期联编。要实现静态联编,在编译阶段就必须确定程序中的操作调用(如函数调用)与执行该操作代码间的关系,确定这种关系称为束定,在编译时的束定称为静态束定。静态联编对函数的选择是基于指向对象的指针或者引用的类型。其优点是效率高,但灵活性差。

 

//例1:静态联编

#include"iostream.h"

class A
{
    

public: void f()
{
  cout
<<"A"<<"";
} }; class B: public A {
public: void f(){cout<<"B"<<endl;} }; Void main() {
  A
* pa=NULL;   A a;
B b;   pa
=&a; pa->f();   pa=&b; pa->f(); }

 

该程序的运行结果为:A   A

从例1程序的运行结果可以看出,通过对象指针进行的普通成员函数的调用,仅仅与指针的类型有关,而与此刻指针正指向什么对象无关。要想实现当指针指向不同对象时执行不同的操作,就必须将基类中相应的成员函数定义为虚函数,进行动态联编。

 

2. 动态联编

动态联编是指联编在程序运行时动态地进行,根据当时的情况来确定调用哪个同名函数,实际上是在运行时虚函数的实现。这种联编又称为晚期联编,或动态束定。动态联编对成员函数的选择是基于对象的类型,针对不同的对象类型将做出不同的编译结果。C++中一般情况下的联编是静态联编,但是当涉及到多态性和虚函数时应该使用动态联编。动态联编的优点是灵活性强,但效率低。

动态联编规定,只能通过指向基类的指针或基类对象的引用来调用虚函数,其格式为:指向基类的指针变量名->虚函数名(实参表)或基类对象的引用名.虚函数名(实参表)

实现动态联编需要同时满足以下三个条件:

①    必须把动态联编的行为定义为类的虚函数。

②    类之间应满足子类型关系,通常表现为一个类从另一个类公有派生而来。

③    必须先使用基类指针指向子类型的对象,然后直接或者间接使用基类指针调用虚函数。

//例2:动态联编

#include"iostream.h"

classA

{
public:

Virtual void f()//虚函数

{cout<<"A"<<"";}};

class B:public A

{
public:
Virtual void f()//虚函数
{cout<<"B"<<endl;}

};

void main()

{ 
A* pa=NULL;

A a; B b;
pa=&a;
pa->f();
pa=&b;
pa->f();

}

 

该程序的运行结果为:A  B

从例2程序的运行结果可以看出,将基类A中的函数f定义为虚函数后,当指针指向不同对象时执行了不同的操作,实现了动态联编。

3. 动态联编分析

 动态联编要求派生类中的虚函数与基类中对应的虚函数具有相同的名称、相同的参数个数和相同的对应参数类型、返回值或者相同,或者都返回指针或引用,并且派生类虚函数所返回的指针或引用的基类型是基类中虚函数所返回的指针或引用的基类型的子类型。如果不满足这些条件,派生类中的虚函数将丢失其虚特性,在调用时进行静态联编。

例3:通过指向基类的指针来调用虚函数

#include"iostream.h"

Class base

{

public:

virtual void fun1(){cout<<"base fun1"<<endl;}

virtual void fun2(){cout<<"base fun2"<<endl;}

void fun3(){cout<<"base fun3"<<endl;}

void fun4(){cout<<"base fun4"<<endl;}

};

Class derived:public base

{
    public:

Virtual void fun1(){cout<<"derived fun1"<<endl;}

Virtual void fun2(intx){cout<<"derived fun2"<<endl;}

Virtual void fun3(){cout<<"derived fun3"<<endl;}

Void fun4(){cout<<"derived fun4"<<endl;}};

Void main()

{
    base*pb;

derive dd;

pb=&d;//通过指向基类的指针来调用虚函数
pb->fun1();
pb->fun2();
pb->fun3();
pb->fun4();

}

 

该程序的运行结果:

Derived fun1 base fun2 base fun3 base fun4

 

本例中函数fun1在基类base和派生类derived中均使用了关键字virtual定义为虚函数,并且这两个虚函数具有相同的参数个数、参数类型和返回值类型。因此,当指针pb访问fun1函数时,采用的是动态联编。函数fun2在基类base和派生类de-rived中定义为虚函数,但这两个虚函数具有不同的参数个数。函数fun2丢失了其虚特性,在调用时进行静态联编。函数fun3在基类base中说明为一般函数,在派生类derived中定义为虚函数。在这种情况下,应该以基类中说明的成员函数的特性为标准,即函数fun3是一般成员函数,在调用时采用静态联编。函数fun4在基类base和派生类derived中均说明为一般函数,因此基类指针pb只能访问base中的成员。

 

 

例4:通过基类对象的引用来调用虚函数

#include"iostream.h"

Class CPoint

{
    public:

CPoint(doublei,doublej){x=i;y=j;}

Virtual double Area(){
   return 0.0;}

private:

double x,y;

};

Class CRectangle:public CPoint

{
    public:

CRectangle(double i,double j,double k,double l);

Double Area(){
    return w*h;}

private:

double w,h;

};

CRectangle::CRectangle(double i,double j,double k,double l):CPoint(i,j)

{  w=k;h=l;  }

Void fun(CPoint &s)

{  cout<<s.Area()<<endl;  }//通过基类对象的引用来调用虚函数

Void main()

{

CRectangle rec(3.0,5.2,15.0,25.0);

fun(rec);

}

 

该程序的运行结果为:375

 例4中的成员函数Area在基类CPoint中使用了关键字virtual定义为虚函数,在派生类CRectangle中定义为一般函数,但是进行了动态联编,结果为15*25即375。这是因为一个虚函数无论被公有继承多少次,它仍然保持其虚特性。在派生类中重新定义虚函数时,关键字virtual可以写也可不写,但为了保持良好的编程风格,避免引起混乱时,应写上该关键字。

4. 小结

    从以上四个例子中可以看出:虚函数是实现多态的基础,是实现动态联编的必要条件之一。动态联编要靠虚函数来实现,虚函数要靠动态联编的支持。两者相辅相成,缺一不可。

转载于:https://www.cnblogs.com/yep3575/p/3499322.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/109943.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 思科路由器配置命令(一)

    思科路由器配置命令(一)一 路由器基本配置命令 R1 gt enable 进入特权模式 R1 disable 退出特权模式 R1 gt R1 configureter 进入全局配置命令 R1 config noipdomain lookup 关闭域名解析 R1 config hostnameSW1 更改主机名为 SW1R1 config enablepasswo 配置进入特权模式的密码 R1 config interfa

    2025年7月3日
    5
  • flume和kafka区别

    flume和kafka区别kafka和flume都是日志系统,kafka是分布式消息中间件,自带存储;flume每一部分都是可以定制。kafka更合适做日志缓存,flume数据采集部分做的很好,可以定制很多数据源,减少开发量。kafka和flume都是日志系统,kafka是分布式消息中间件,自带存储,提供push和pull存取数据功能。flume分为agent(数据采集器),collector(数据简单处理和写入),storage(存储器)三部分,每一部分都是可以定制的。比如agent采用RPC(Thri.

    2022年6月23日
    27
  • eclipse import导入报错_project2016安装出错

    eclipse import导入报错_project2016安装出错【转载请注明出处:http://blog.csdn.net/zjbpku/article/details/7370347】本文只介绍方法不解释原因:0,如果R.layout.main有误,将importandroid.R删除一,删除gen文件,会自动生成一个新的gen(Project下的BuildAutomatically要选上)二、右键点击有误文件,选择BuildP

    2022年9月1日
    4
  • html asp 简单 登录系统,asp用session做登录页面

    html asp 简单 登录系统,asp用session做登录页面asp写登陆页面。利用session记录用户信息。先建好数据库连接文件,命名为conn.asp。1:登录页面login.htm登录用户名:密码:2:登录检测页面go.aspasp网页登录后显示用户名,怎样用session就好像在图片欢迎登录中间显示每个登录不同的用户名asp中SESSION具体用法ASP中session的用法。CSS布局HTML小编今天和大家分享ASP中…

    2022年7月15日
    17
  • python京东自动签到领金豆_docker 京东自动签到

    python京东自动签到领金豆_docker 京东自动签到项目紧张的忙完了,早上签到时突然想到自动签到~~’人生苦短,我用python’网上看了下,很简单。对于小白来说,主要难度是环境的搭建。主要用到:1selenium模拟浏览器2chromedriver(chrome驱动)上面网友已经实现飞猪京东签到,依葫芦画瓢嘛,实现了苏宁易购的签到。备注:只是很简单签到代码,没有登录的滑动签到的校验码(第一次登录签到)参照上面的,自己实现了苏宁易购的…

    2022年9月18日
    2
  • TCP的拥塞控制_假设tcp拥塞控制算法中

    TCP的拥塞控制_假设tcp拥塞控制算法中在某段时间,若对网络中某一资源的需求超过了该资源所能提供的可用部分,网络性能就要变坏,这种情况就叫做网络拥塞。在计算机网络中数位链路容量(即带宽)、交换结点中的缓存和处理机等,都是网络的资源。若出现拥塞而不进行控制,整个网络的吞吐量将随输入负荷的增大而下降。当输入的负载到达一定程度吞吐量不会增加,即一部分网络资源会丢失掉,网络的吞吐量维持在其所能控制的最大值,转发节点的缓存不够大这造成分…

    2022年4月20日
    57

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号