【教学课件】第四章面向对象程序设计语言.ppt
第四章 面向对象程序设计语言,4.1 Smalltalk语言对象的思想最早源于人工智能研究,60年代末描述智能对象的框架(frame)即封装了许多槽(slot),槽既可以是属性(数据)也可以是行为(操作)和(约束)。但最早见诸文献是sketchpad提到的OO图形学(1963)。60年代挪威的Dahl和Nyard为模拟系统研制了SIMULA-67语言,首先提出封装的类和动态生成实例对象的概念。60年代末,美国犹他大学Alan Kay到Xerox公司PaloAlto研究中心参加了Dynabook项目。该项目的硬件是Star(个人机的前驱)软件是Smalltalk。1972年Dan Ingalls完成Smalltalk-72第一个实用版,以后又经过-76-80两次改进,Smalltalk-80成为向外发行的正式版本。,4.1 Smalltalk语言对象的思想最早源于人工智能研究,60年代末描述智能对象的框架(frame)即封装了许多槽(slot),槽既可以是属性(数据)也可以是行为(操作)和(约束)。但最早见诸文献是sketchpad提到的OO图形学(1963)。60年代挪威的Dahl和Nyard为模拟系统研制了SIMULA-67语言,首先提出封装的类和动态生成实例对象的概念。60年代末,美国犹他大学Alan Kay到Xerox公司PaloAlto研究中心参加了Dynabook项目。该项目的硬件是Star(个人机的前驱)软件是Smalltalk。1972年Dan Ingalls完成Smalltalk-72第一个实用版,以后又经过-76-80两次改进,Smalltalk-80成为向外发行的正式版本。,4.1.1 Smalltalk系统,语言核心(Kernel)程序设计系统程序设计范型(Paradigm)用户界面模型(User Interface Model),4.1.2 用户界面模型,系统工作空间(System WorkSpace)工作空间(WorkSpace)系统副本(System Transcript)项目(Project)两种图形编辑窗(Form和Bit),系统浏览器(System Browser)窗,用户就是按浏览窗中显示的模板填写程序。,4.1.3 语言核心,(1)保留字只有五个nil,true,false,self,super(2)字面量字符字面量/数字面量/符号字面量/数组字面量(3)限定符和特殊符号#(),;::=或()(4)变量实例变量/类变量/临时变量/全局变量/汇聚变量/参数,(5)消息表达式与语句,消息表达式的一般格式是:对象 选择子 参数Smalltalk的消息表达式有三种:单目的 不带参数 tree class 消息class 发向tree,得到tree的类。0.3 sin 消息sin 发向0.3,得sin(0.3)Array new 消息new 发向Array,创建-Array 的实例,对象,选择子-参数,双目的 3+4 消息+带参数4发向对象3,得对象7。100 50 消息带参数50发向对象100,得(100,50)(sum/count)*reserve amount 双目,括号优先 单目优先 双目,关键字消息表达式用关键字(带有:的选择子)描述的双目表达式,也是自左至右释义。anArray at:3 put:100finances totalSpentOn:food赋值 变量在不同时间可赋以不同对象,任何表达式加上赋值前缀 quantity19.namechapter 1。foo array at:4。数组第4元素与foo同名,块表达式:x:yBicPen goto:xy:x:yBicPen goto:xy value:100 value:250 BicPen goto 100 250 aBlock aBlockThis is a String displayAt:500 500.Display white.aBlock value,(6)控制结构 条件选择一般形式是:布尔子表达式 ifTrue:真块执行 ifFalse:假块执行“可以不出现”如:numberlistSize whileFalse:list at:index put:0。indexindex+1,(7)消息/方法消息模式|临时变量|语句组 nawAt:initialLocationnewBox newBoxself new.newBox setLoc:initiaPLocation tilt:0size:100 scribe:pen new.newBox show.setLoc:newLoc tilt:newTilt size:newSize seribe:newScribe LocnewLoc.titlnewTilt.sizenewSize.scribe new Scribe,Smalltalk是编译解释执行的,Smalltalk源程序经编译器得到虚映象(Virtual image),虚映象由字节代码中间语言编写,由Smalltalk虚机解释执行。相应的文件系统管理三种文件:源文件、变更文件、映象文件。由于Smalltalk是交互式的,被编译的方法在执行期间出了问题要反应到源程序,则要对映象文件施行反编译(decompliation)Smalltalk的虚机是一个新软件,它有三个功能部分:存储管理器虚映象解释器基本例程 用汇编码写出的底层方法实现,4.1.4 Smalltalk文件系统与虚机,4.1.5 Smalltalk程序设计范型,程序设计在类的层次上进行,由类静态(于工作空间指明向类发出消息)或动态(方法运行时)生成实例对象。每个对象当接受某消息并执行其方法的消息表达式时都是在自向其它对象发消息。,4.1.5.1 一个简单的Smalltalk程序,统计字母出现频率s f“定义了两个临时变量”sPrompter prompt:enter line default:.“s是Prompter的实例,将关键字表达式的结果束定于s”“意即输入一行字符串,若不输入,S为空串”fBag new.“f是Bag的实例”s do::cc isLetter ifTure:f add:c asLowerCase“s在Prompter中找方法do:的模式,若找不到,找prompter的”“父类直到Object.C是块变量,意从S中拿出某字符,isLetter”“是消息模式,判C是否字符,若为真执行内块”。“内块中f找add:消息模式,从Bag直至上层父类,找到先执”“行右边子表达式”。c asLowerCase是单目表达式,同样要在Prompter中找asLowerCase匹配,也是不成向上找。它返回是“第k个”小写字母,add:把它发送到对象f的第k个位置上并与原数相加。f“返回f中的值”.这个程序一共四句。如果掀鼠标使菜单项doit工作并输入:“Smalltalk is a programming Language for developing soluions to both simple and complex problem.”则输出的f值是:7 1 1 2 4 1 5 1 5 1 7 4 4 7 3 3 6 3 2 1 a b c d e f g h i k l m n o p r s t u v,例 字频统计对比程序Pascal SmalltalkPROGRAM Frequency“无消息模式方法,宜写算法”CONST Size=80;VAR s:stringsize;|s c f k|k,i:Integer;“定义了四个临时变量”c:Char;f:ARRAY1.26 OF Integer;fArray new:26.BEGIN“f是Arrey实例长度26”Writeln(enter line);sPrompter ReadIn(s);prompt:enterline FOR i:=1TO 26 DO default:.fi:=0;“S是Prompter的实例,装输入字串”FOR i:=1 To size DO 1 to:26 do:I|f at:I put:0.BEGIN 1 to:size do:I|c:=aslowerCase(si);c(s at:i)asLowerCase.if isLetter(c)THEN c isLetter ifTrue:BEGIN kc asciiValue k:=ord(c)-ord(a)+1;-a asciiValue+1.fk:=fk+1 f at:k put:(f at:k)+1 END END;.FOR i:=1 To 26 DO f Write(fi,)END.,4.1.5.2 类协议,4.1.5.3 一个完整的Smalltalk程序,家庭财务帐目 建立全部流水帐类,直接挂在Object上 class name FinancialHistory superclass Object instance variable names caseOnHand incomes expenditures category Financial Tools class method initialBalance:amount“建立流水帐本初始为amount(元)”super new setinitialBalance:amounT new“建立流水帐本初始为0(元)”super new setinitialBalance:0 instance method receive:amount from:source incomes at:source put:lself total ReceivedFrom:source)+amount.“从来源source接收到的钱数,因而手头现金增加”.cashOnHandcashOnHand+amount.incomes changed spend:amount for:reason“为事由reason支付的钱数,因而手头现金减少。”expenditures at:reason put:(self totalSpentFor:reason)+amount.,cashOnHand cashOnHand-amount.expenditures changed CashOnHand“回答当前手头现金”cashOnHand expenditures“回答支出细目”expenditures incomes“回答收入细目”incomes totalReceiveFrom:source“回答自source收钱总数”(incomes includesKey:source)ifTrue:incomes at:source ifFalse:0 totalSpentFor:reason“回答在reason项上总支出”(expenditures includesKey:reason)ifTrue:expenditures at:reason ifFalse:0 private SetlnitialBalance:amount“实例变量初始化”cashOnHandamount.incomesDictionary new.expendituresDictionary new,Smalltalk at:#HouseholdFinances put:nil.HouseholdFinancesFinancealHistory initialBalance:1560 HouseholdFinances spend:700 for:rent.NouseholdFinances spend:78.53 for:food.HouseholdFinances receive:820 from:pay.HouseholdFinances receive:22.15 from:interest.HouseholdFinances spend:135.65 for:utilities.HouseholdFinances spend:146.14 for:food.,4.1.6 Smalltalk程序设计系统,在Smalltalk中,系统支持程序也是作为类挂在Object之下,包括算术运算、数据和控制结构的实现、输入/出、随机数生成器等。有一些类是辅助程序设计过程的,语法分析器、编译器、解释器、反编译器这些对象的方法都有源代码,目标码两种形式。还有一些对象表示类和方法的结构,以便程序员追踪系统。还有将方法和向其发消息的对象联结起来的对象.这些对象统称环境(contexts)类似其他语言实现中的堆栈帧和活动记录。,4.1.7 Smalltalk的对象、类、方法的实现 类的存储,实例对象的存储 实例对象只存放数据,其存储格式如下图:,活动记录 环境部分 指令部分 发送者部分,4.2 面向对象的发展与概念,为什么需要面向对象?OO语言的发展面向对象的基本概念,重用的问题,实践中人们认识到重用已有开发结果的重要性,提出了软件重用的概念最早的重用单元是子程序,如Fortran的子程序库子程序是纯粹的过程抽象,基于子程序的重用有很大局限性模块是更合适的重用单元,因为模块可以包装任何功能,更灵活重用中有一种常见情况:软件开发中遇到的新问题常与解决过的问题(可以重用的库提供的功能)类似,但又不完全相同已有模块的功能与需要有差异,无法以其“现有”形式直接使用如果模块功能的改变只能通过修改源代码的方式进行,程序员就只能拷贝这个模块的源代码,深入研究后再设法修改,以满足新需求但问题是有没有可以使用的源代码?常常没有:模块可能是购入的,提供商不提供源代码模块可能是过去的遗产,源代码已经丢失或部分缺失,模块和程序组织,常规的程序单元缺乏弹性,定义好的子程序/模块都是固定功能的实体,难以提供“定制”的方式部分地改变功能以满足实际需要的变化通过模块定义的抽象数据类型是相互独立的,不同模块之间无任何关系而实际情况中,常常需要定义和使用一些相互有关的类型,可能需要把它们送给同一个函数/过程去处理,以同样方式存储变体和联合机制就是为了迎合这方面的需要,但它们没有类型安全性,且未能提供解决类似问题的统一框架,难用于应付更复杂的情况支持相关类型,可能给程序的结构组织带来新的可能性如何在抽象数据类型的框架中提供这一类功能,也是需要解决的问题面向对象的概念在这些方面都能发挥很大的作用面向对象(Object-Oriented)的方法和程序技术,为基于模块(一个类也可以看作一个模块)的重用问题提供了一条解决途径。,OO发展史,OO技术和思想中的一个基本方面是数据和操作的封装这方面的基本想法:一组数据与关联之上相关的操作形成一个对象。其内部数据构成对象的状态,操作确定对象与外界交互的方式OO并不是从模块化程序设计发展出来的,它有自己的发展历程OO的思想与模块化的思想是并行发展,一直相互影响、相互借鉴Simula67是OO概念的鼻祖,其设计目标是扩充Algol60,以更好地支持计算机在模拟方面的应用。1960年代在挪威计算中心设计和实现,主持其工作的Ole-JohanDahl和KristenNygaard获得2001年图灵奖OO的三个基本要素:封装、继承和动态方法约束都源于Simula类的概念源自Simula,其设计中提出用类定义把一组操作与一组数据包装起来。Simula的这些重要想法是模块概念和OO的起源Simula只提供了基本封装,并没有对封装的保护,也没有信息隐藏,OO发展史,软件实践也需要OO的思想,并逐渐开发了相关的支撑技术,包括:封装的思想在面向模块的语言里发展,提出了许多重要概念和想法,如作用域规则,开的或者闭的作用域界面与实现透明类型与隐晦类型,访问控制,等等数据驱动的程序设计技术:将计算功能(子程序)约束于程序里处理的数据(结构),使我们在程序里可以从数据对象出发去启动相应的计算过程在一些非常规的语言(如函数式语言)里,可以通过引用的概念提供函数/过程与数据之间的约束常规语言(如C)引进了指向函数的指针,在实现数据驱动程序设计的过程中起到了重要作用,也成为面向对象语言实现的技术基础,OO发展史,继承和动态约束等被Smalltalk发展,形成目前OO的基本概念框架程序里以类的方式定义各种数据抽象类可以通过继承的方式扩充新功能,这样定义的新类(子类,派生类)自动继承已有类(基类,超类,父类)的功能对象是类的实例,是程序运行时的基本数据单元派生类的对象也看作是原有基类的对象,可以当作基类的对象使用(子类就是子类型,Liskov代换原理,2008年图灵奖)类定义了对象的状态成分(数据成员)和一组相关操作(称为方法)方法调用总是针对某个对象进行的,将方法调用看作是给相应对象送一个消息,对象通过执行相应操作的方式对消息做出响应对一个消息执行什么方法,由接收消息的对象的类型确定(根据该对象所属的类确定,这就是动态约束)计算,就是一组对象相互通讯的整体效果(对计算的另一种看法),OO发展史,Smalltalk还有一些独特的东西:变量采用引用模型,变量无类型,可以引用任何对象语言里的一切都是对象:类也是对象,通过给类送new消息的方式要求创建类的实例各种控制结构也是通过消息概念建立的条件和逻辑循环是逻辑对象对特定消息的响应枚举循环是整数对象对特定消息的响应采用单根的类层次结构,以类Object作为所有类的超类提供了块(block)的概念,作为控制结构的抽象机制提出了容器的概念,开发了一个功能丰富的类库与程序开发环境的紧密结合,并开发了GUI的基本概念和相关技术Smalltalk经过72、76发展到Smalltalk80,其概念和结构已臻成熟,OO发展史,随着Smalltalk的成功,人们看到了OO的潜在威力许多人开始研究如何把OO概念有效集成到常规语言里,提出了一批已有语言的OO扩充和许多新OO语言,如Object-Pascal、Object-C等其中前期最成功并得到广泛应用的是C+。C+在OO概念的广泛接受和应用方面功不可没(具体理由见后面讨论)。原因:在面向对象和高效程序之间取得较好的平衡OO概念与常规语言的合理集成(在当时),支持数据抽象和面向对象的系统设计和程序设计,支持多泛型程序设计的结合,使与数据抽象和OO有关的许多新概念和新技术逐渐被实际软件工作者接受随后是OO分析、OO设计和基于OO的软件开发等等后来的其他成功语言包括Java,微软提出C#,等等出现了一些基于对象的脚本语言,如Python,Ruby等现在,面向对象的开发已经成为一种主流的软件开发技术,面向对象的基本概念,面向对象的基本概念:在面向对象语言里定义数据抽象的基本定义机制是类,在一个类里可以定义数据成员和子程序成员(称为方法)封装是数据抽象和模块化的概念,与面向对象的概念并没有必然关系,但封装有助于更好发挥面向对象机制的作用(实在的)类被看作类型,可以用于生成(定义)实例,称为对象已有的类可以作为定义新类的基础(基类、超类)可通过继承方式定义新类(子类,派生类),子类继承基类的行为子类可以修改基类已经定义的行为,或者增加所需的新行为把子类看作是子类型(通常),如果D是B的子类,那么:若o是D类型的对象,那么o也看作是B类型的对象若变量x可以引用B类的对象,那么它也可以引用D类的对象,面向对象的基本概念,继承有两方面作用1.建立类型之间的层次关系2.重用基类的行为(代码和数据描述)对于面向对象的行为而言,前一方面的功能更为重要类中的子程序成员称为方法,方法需要通过具体的对象调用在运行中调用方法时,实际调用的方法由作为调用出发点的那个对象的类型确定的(动态约束)动态约束是实现面向对象行为的关键它为面向对象的机制提供了模块机制所不具有的弹性,使新的功能扩充可以比较自然地结合到已有的操作过程里理解动态约束是理解面向对象的关键,动态约束的高效实现也是面向对象语言的实现的关键,面向对象的语言,虽然基本框架类似,不同面向对象语言之间也存在很大差异:基本问题:采用什么样的对象模型采用单根的类层次结构,还是任意的类层次结构?提供那些继承方式?例如C+里提供了三种继承方式允许多重继承?还是只允许单继承?是否提供丰富完善的访问控制机制?采用基于继承的模型,还是基于指派的模型基于类的模型,还是基于对象或原型的模型(如JavaScript)对象本身的独立性(是否允许不属于任何一个类的对象)类本身是不是对象?,面向对象的语言,其他情况:是不是追求“纯粹”的面向对象语言?Smalltalk尽可能追求“面向对象”理想,完全是重新设计的新语言Java是接近理想的语言,但希望在形式上尽可能靠近常规语言C+设法在支持系统程序设计的过程性语言C上“扩充”支持面向对象的机制,是一种多范型语言,支持多种程序设计方式另外的一些语言(如Ada)采用可能很不同的方式支持面向对象的程序设计,这里不准备详细介绍采用值模型还是引用模型。从本质上说,只有采用引用模型才能支持方法的动态约束,因此大多数面向对象语言采用引用模型C+采用值模型,可以创建静态对象或栈对象,但只有通过对象引用或指向对象的指针才能实现面向对象的动态约束行为Java只能把OO功能应用于用户定义类型,基本类型采用值模型,面向对象的语言,是否允许静态对象或者堆栈对象(自动对象)?多数面向对象语言只支持堆对象(通过动态存储分配创建的对象)C+支持静态对象和自动对象,这种设计是希望尽可能借助于作用域规则来管理对象,避免依赖自动存储管理系统(GC)为在这种环境下编程,人们开发了许多利用自动对象的对象管理技术,如句柄对象,对象的“创建即初始化”技术等是否依赖自动废料收集(GC)。由于OO程序常(显式或隐式地)创建和丢弃对象,对象之间常存在复杂的相互引用关系,由人来完成对象的管理和回收很困难。大多数OO语言都依赖于自动存储回收系统GC的引入将带来显著的性能损失,还会造成程序行为更多的不可预见性(GC发生的时刻无法预见,其持续时间长短也无法预计)Java等许多语言都需要内置的自动废料收集系统C+是例外,其设计目标之一是尽可能避免对自动存储回收的依赖,以支持系统程序设计,提高效率,减少运行时间上的不确定性,面向对象的语言,是否所有方法都采用动态约束?动态约束很重要,但调用时会带来一些额外的开销,如果需要调用的方法能够静态确定,采用静态约束有速度优势大部分语言里的所有方法都采用动态约束C+和Ada提供静态约束(默认)和动态约束两种方式一些脚本语言也支持面向对象的概念。例如,Ruby是一个纯面向对象的脚本语言,其中的一切都是对象,全局环境看作一个匿名的大对象,全局环境里的函数看作这个对象的成员函数。它还有另外一些独特性质JavaScript支持一种基于对象和原型的面向对象模型。其中没有类的概念,只有对象。对象的行为继承通过原型获得,面向对象的语言,人们还提出了许多与面向对象机制有关的新想法和模型许多新近的脚本语言提供了独特的面向对象机制:例如基于对象原型(而不是类)的OO模型在基于类的模型中允许基于对象的行为覆盖(可修改个别对象的行为)等等总而言之,虽然今天面向对象的模型和语言已成为主流程序设计方法和主流程序语言,但是这类语言还远未成熟,还正在发展和研究中许多语言的OO机制非常复杂,实际还不断提出一些新要求,使一些OO语言在发展中变得越来越复杂如何提供一集足够强大,而且又简洁清晰的机制支持OO的概念和程序设计,还是这个领域中需要继续研究的问题OO语言有关的理论研究还处在起步阶段,也是本领域不成熟的标志,OO语言需要提供的新机制,定义类的语言机制(语言提供特殊的描述结构)描述或定义对象的机制继承机制,描述类之间的继承关系。可能定义继承关系的性质(如C+里的类继承有public、protected和private三种方式)与对象交互的机制(方法调用,消息传递)初始化新对象的机制(最好能自动进行,避免未初始化就使用的错误)类类型对象的动态转换机制(转换对一个具体对象的观点)控制类成员的访问权限的机制对象销毁前的临终处理机制(最好能自动进行)对象的存储管理机制可能还有其他机制:运行中判断对象的类属关系的机制、自反等等,4.3 面向对象语言的基本特征与实现,P.Wegner总结了OO语言的发展,给出以下图示澄清了概念:封装 对象(数据和操作)局部性、可维护性抽象+类 概括描述、简单性继承+类体系 可重用性多态 重载、类属 可扩充性动态束定 面向对象 可交互性 基于对象的语言 基于类的语言 面向对象语言 Ada 83,Actor CLU Smalltalk、Eiffel simula 67 C+,Ada 95,Java,OO程序,先看一点OO程序,复习一下基本OO程序的特征这里看一段定义了几个类的C+代码,定义list_node类,用于实现带头结点的双向循环链接表,每个结点里有一个域指向表头结点,OO程序,定义list_node类,用于实现带头结点的双向循环链接表,每个结点里有一个域指向表头结点,OO程序,定义一个list类,注意:header是个list_node,定义的是有头结点的循环链表,OO程序,通过继承定义queue类。(只是作为示例),OO程序,还可以定义通用的容器类:基本容器类没有具体数据域,不保存具体类型的元素,只实现容器操作,如:一些基本判断谓词,插入删除等等通过继承实现存储具体类型的元素的具体容器,每个结点里有一个域指向表头结点,OO程序,派生的int表结点类,使用这种int表的问题:如果需要访问结点的数据内容,必须对取出的结点做强制,通用的表结点类,面向对象特征的实现,实现面向对象的语言,需要考虑它的几个标志性特征的实现封装是一种静态机制,如C+/Java一类语言的各种访问控制机制也是静态的,都可以通过在符号表里记录信息,在编译中检查和处理方法的实现与以模块为类型时局部子程序的实现一样。由于每个方法调用有一个调用对象,因此方法需要一个隐含指针,被调用时指向调用对象,所有对该对象的数据成员的访问都通过这个指针和静态确定的偏移量进行许多语言以这一指针作为一个伪变量,称为this或者self,通过这种指针访问调用对象,方式上与通过指针访问普通结构一样实现面向对象语言的关键是两个问题:继承的实现,使派生类型的对象能当作基类的对象使用动态约束的实现,能够从(作为变量的值或者被变量引用的)对象出发,找到这个对象所属的类里定义的方法下面讨论实现的一些具体问题,封装,封装是一种静态机制,仅仅在程序加工阶段起作用,有关情况与模块机制类似,在加工后的程序里(可执行程序里)完全没有关于封装的信息不同语言里对类的访问控制可能不同:作为“开模块”(允许以特定方式任意访问类成员)作为“闭模块”(凡是没有明确声明可访问的都不可访问)对基本封装机制的扩充是引进进一步的控制C+引进成员的public、protected和private属性,提供细致的访问控制C+还允许定义派生类的不同继承方式,控制对基类成员的访问:public继承protected继承,使基类的public成员变成派生类的protected成员private继承,使基类的所有成员变成派生类的private成员一些新语言借鉴了C+的这方面思想,可能结合另外一些想法,静态域和静态方法,许多面向对象语言的类里可以定义静态域和静态方法C+/Java允许类里定义静态数据域Smalltalk把普通的对象域称为实例变量,表示在这个类的每个实例里都有这些成分的一份拷贝;把静态数据域称为类变量类的静态数据域并不出现在实例对象里,它们是类封装的静态数据成分,提出具有静态生存期,在类的作用域里可直接访问。类外能否访问由语言确定(提出有与其他成员一样的访问控制)静态方法和静态域的一些情况:类的静态数据成员可以在静态区实现,在程序运行之前静态分配,在程序的整个执行期间保持其存储类的静态方法可访问静态数据成员,其他方法也可以访问静态数据成员可以把静态数据成员看作本类的所有对象共享的信息类对象可以通过静态数据成员交换或者共享信息,静态域和静态方法,静态成员是静态创建的,其初始化在程序开始执行前完成(或者在语言定义的适当时刻完成),只做一次静态成员的初始化中不能调用类的普通方法(因为没有对象)静态方法相当于普通子程序,只是具有类封装(类作用域)。特点:没有调用对象不能引用this/self,不能引用类定义的普通数据成员(如Smalltalk里不能引用实例变量),只能引用本类的静态数据成员通常采用某种基于定义类的语法形式调用仅有静态数据成员和静态方法的类,相当于一个简单模块提供模块的内部状态,可以通过所提供的方法修改状态不能生成有用的实例(生成的是空实例,没有局部的实例状态)静态数据成员的静态方法的封装,可能定义内部数据和操作,对象和继承:数据布局,继承关系的数据部分通过对象的适当存储布局实现对象的实际表现就是数据成员的存储假定B是一个类,有自己的数据成员D是B的派生类,增加了数据成员。D类对象的前部仍是B类的所有成员,扩充的成员排在后面在D类对象里,所有B类成员相对于对象开始位置的偏移量与它们在一个B类对象里的偏移量相同这样,D类对象就可以作为B类对象使用,B类里的方法能正确操作,它们只看属于B对象的那部分D类里的方法既可以使用对象中的B类数据成员,也可以使用对象里的D类数据成员用D类对象给B类对象“赋值”(值copy,或者值语义时)会产生“切割”现象,D类数据成员不能拷贝,B类的数据成员,B类的数据成员,D类新增的数据成员,B类的对象,D类的对象,初始化和终结处理,对象可能具有任意复杂的内部结构要求创建对象的程序段做对象初始化,需反复描述,很麻烦,易弄错对象可能要求特殊的初始化方式和顺序,对象的使用者难以贯彻始终继承使对象的初始化更复杂化,因为需要正确初始化继承来的数据成员为更容易处理对象初始化的问题,OO语言通常都提供了专门的机制,在对象创建时自动调用初始化操作保证新创建对象具有合法的状态。自动调用非常有意义,可以避免未正确初始化造成的程序错误现在常把对象初始化看作调用一个称为构造函数(constructor)的初始化子程序,它(们)在对象的存储块里构造出所需要的对象语言通常支持参数化的初始化操作,以满足不同对象的需要。对象创建可能有多种需要,为此C+/Java等都支持一个类有多个不同的构造函数,初始化和终结处理,如果变量采用引用语义,所有(值)对象都需要显式创建,有明确的创建动作。这样很容易保证在存储分配之后调用构造函数如果变量是值,为保证初始化,语言需要对变量创建提供特殊语义,要求变量创建包含隐式的构造函数调用对象初始化必须按一定的顺序进行对象内部的基类部分必须在派生类部分之前完成初始化,因为派生类新增的数据成员完全可能依赖于基类成员的值数据成员本身也可能是某个类的对象,在执行整体对象的构造函数的过程中,就需要执行这些对象成员的构造函数这种构造规则是递归的,语言必须严格定义对象的构造顺序如果变量采用值语义(例如C+),在进入一个作用域的过程中,就可能出现许多构造函数调用进入作用域可能是代价很大的动作,初始化和终结处理,在对象销毁之前,可能需要做一些最后动作(终结处理),例如释放对象所占用的各种资源,维护有关状态等忘记终结处理,就可能导致资源流失,或者状态破坏有些OO语言提供终结动作定义机制,销毁对象前自动执行所定义动作C+采用值语义,终结动作以类的析构函数的形式定义:类变量是堆栈上的对象,在其作用域退出时,自动调用它们的终结动作堆对象需要显式释放,释放之前恰好应该执行终结动作,易于处理采用引用语义的语言(如Java),通常并不提供销毁对象的显式操作(以防悬空引用),对象销毁由GC自动进行有了GC,对终结动作的需求大大减少,终结动作由GC自动进行执行终结动作的时间不可预计,出现了(时间和顺序的)不确定性对象关联和GC顺序的不确定性使终结动作很难描述,静态和动态约束的方法,OO语言里的方法调用通常采用x.m(.)的形式,其中x是一个指向或者引用对象的变量m是x的定义类型(类,假定为B)的一个方法问题:x.m(.)所调用的方法何时/根据什么确定?两种可能性:根据变量x的类型(在程序里静态定义)确定(静态约束)根据方法调用时(被x引用/指向)的当前对象的类型确定(动态约束)由于x可能引用(指向)B类或其任何子类的对象,因此同为这个方法调用,不同的执行中实际调用的完全可能是不同的方法所有OO语言都支持动态方法约束(否则就不是OO语言),多数以它作为默认方式。少数语言同时也支持静态约束的方法,如C+、Ada等C+把动态约束的方法称为虚方法(virtual方法),而且以静态约束作为默认方式。这种设计与它的C基础有关,静态约束的实现,调用静态约束的方法,实现方式就像是调用普通子程序(过程/函数),唯一不同之处就是需要传入一个指向调用对象的指针在符号表里,每个类的记录项都包含了一个基类索引,依靠这个索引形成的基类链就可以静态(编译时)完成静态约束的方法的查找:1.首先在变量所属的类(静态已知)里查找(查找符号表)。如果在这里找到了所需要的方法,就生成对它的调用;如果不存在就反复做下一步2.转到当前类的基类里去查找相应方法,如果找到就生成对它的调用;如果找不到就继续沿着基类链上溯查找3.如果已无上层基类,查找失败。报告“调用了无定义的方法”错误所有对静态约束的方法的调用都可以静态(编译时,一次)处理运行时的动作与一般子程序调用完全一样,没有任何额外运行开销如果语言允许静态约束的方法,采用静态约束可以提高效率。静态约束的方法还可以做inline处理,方法的动态约束,B类里定义了一个一般性操作tem,对所有B类对象都有价值tem中调用了一个特殊操作sp,该操作可能因子类不同而不同子类D覆盖操作sp后,仍能正常地调用操作tem,而且其中对sp的调用能调用到D类里新的操作定义,这是OO程序设计里最重要的东西:这一特征使新类给出的行为扩充(或修改)可以自然地融合到已有功能里,包括放入已有的操作框架里(这个例子就是),动态约束的实现:一般模型,对最一般的对象模型,运行中调用动态约束的方法时要做一次与编译时处理静态约束方法一样的查找,这种查找可能非常耗时为完成这种方法查找:每个类需要有一个运行时表示(把类也作为程序对象),类表示中需要有一个成分是基类索引,还有一个成分是方法表每个对象里必须保存所属类的信息(一个类指针,指向其类)每个动态方法调用都启动一次方法查找。如果找到就调用,找不到就发出一个“messageisnotunderstood”(Smalltalk)动态错误这种方式普遍有效,可以处理具有任何动态性质的对象模型,如动态类层次结构构造和动态方法更新(修改、添加和删除)、动态类属关系等查找的时间开销依赖于继承链的长度和继承链上各个类中方法的个数这种方法的缺点是效率太低。如果所采用的对象模型在动态特性方面有所限制,就可能开发出效率更高的方法,动态约束的实现:受限模型,早期OO语言(包括Smalltalk)都采用功能强大灵活的对象模型在提供了极大灵活性的同时,也带来效率上的巨大开销这也是早期OO语言及其概念难被实际软件工作者接受的最关键原因提高算法效率的最基本途径是限制要解决的问题(对更特殊一些的问题,可能找到效率更高的算法),并设计优化的实现模型对于OO语言,就是要找到一个受限的对象模型,它能比较高效地实现,同时又能满足绝大部分实际OO程序开发的需要常规OO语言中的对象模型有如下特性(足以支持常见程序设计工作):类层次结构是静态确定的每个类里的动态约束方法的个数和顺序都静态确定在这种模型里就可以避免动态方法查找,使方法调用的执行效率接近普通的子程序调用的执行效率(C+和Stroustrup的贡献),动态约束的实现,优化实现模型,其中绝大部分工作都能静态完成:每个类的运行时体现是一个动态约束方法的表(称为虚表,vtable),这是一个指针表,指针指向本类的对象需要调用的方法的代码体虚表的指针按方法在类里的顺序排列,每个方法对应于一个顺序下标,动态约束的实现,在每个对象开头(数据域之前)增加一个指针vt创建对象时,设置其vt指向其所属的类的虚表(运行中始终不变)f是指向F的指针(或引用)调用f-m(.)的实现比