Java基础之面向对象机制(多态、继承)底层实现

一、Java的前世

为什么会产生Java?Java的特点是什么?

从C语言开始讲,C语言是一种结构化语言,模块化编程,便于程序的调试,依靠非常全面的运算符和多样的数据类型,可以轻易完成各种数据结构的构建,通过指针类型更可对内存直接寻址以及对硬件进行直接操作,因此既能够用于开发系统程序,也可用于开发应用软件。其缺点就是封装性弱,程序的安全性上不是很好。C语言的异常处理一般使用setjmp()与longjmp(),在捕获到异常时进行跳转;或者使用abort()和exit()两个函数,强行终止程序的运行。如果要实现多线程,应该要直接操作底层操作系统,语言本身没有封装该机制。

C语言是一门面向过程的语言,所谓面向过程指的是以“事件过程”为中心的编程思想,即按照事件的解决步骤,在函数中一步一步实现。其实,生活中大部分事情都可以用面向过程的思想来解决。然而,当问题的规模变大,且问题中有多个部分是共同的特征时,我们仍然需要对这件事情建立一步一步操作,此时面向过程就显得繁重冗余,因此产生了面向对象的思想。

面向对象的思想是将事件中的一些共同特征抽象出来作为一个对象,一个对象包括属性和方法,比如说一个班级中的同学,大家都拥有姓名、成绩、年龄、兴趣爱好,在操作这些数据的时候,我们只需要将共同的部分抽象出来,然后给每个同学一个对象的实例。如果是面向过程的方法,我们需要为每一个同学定义属性变量,执行某个动作需要定义独立的方法。因此,产生了C++语言。

C++继承自C语言,可以进行面向过程的程序设计,也可以抽象化出对象进行基于对象的程序设计,也可以进行继承、多态为特点的面向对象的程序设计。在C++的面向对象设计中,将数据和相关操作封装在一个类中,类的实例为一个对象。支持面向对象开发的四个特性:封装、抽象、继承、多态。在C++语言中,内存分为堆(程序运行时分配内存)和栈(函数内部声明的变量)两部分,往往需要手动管理内存,通过new,delete动态划分内存并进行内存的回收;类中包含构造函数和析构函数,分别为建立对象和删除对象释放资源。

Java也是一门面向对象的语言,不仅吸收了C++的各种优点,同时摒弃了C++种难以理解的多继承、指针等概念,功能更加强大且简单易上手。其特点:简单、OOP、平台无关性(JVM的功劳);相比于面向对象的语言C++而言,Java JVM的动态内存管理非常优秀。在发展的过程中,逐渐更新了更多强大的功能:XML支持、安全套接字soket支持、全新的I/O API、正则表达式、日志与断言;泛型、基本类型的自动装箱、改进的循环、枚举类型、格式化I/O及可变参数。

在C语言、C++、Java的演化过程中,并不会导致新语言取代旧语言,每种语言按照自身的特点有了自己适合的领域。如追求程序的性能和执行效率,如系统底层开发,就需要使用C++,甚至C语言;安卓开发、网站、嵌入式、大数据技术等领域,一般使用Java语言,由于其安全性、便携性、可移植性、可维护性等,很容易实现多线程,代码的可读性高。

C语言和C++是编译型的语言,而Java是解释型的语言:

  • 编译型语言:程序在执行之前需要一个专门的编译过程,把程序编译成为机器语言的文件。

程序执行效率高,依赖编译器,跨平台性差

  • 解释型语言:程序不需要编译,程序运行时才翻译成机器语言,每执行一次都要翻译一次。

效率比较低,依赖解释器,跨平台性好

Java是静态-强类型语言。

二、多态

多态是同一个行为具有多个不同表现形式或形态的能力。多态就是同一个接口,使用不同的实例而执行不同操作。

一般实现形式:

  • 重载@Overload:同一类种方法名相同,参数不同;返回类型不要求
  • 重写@Override:子类继承自父类的方法重写实现过程,返回值、形参不能变、只能重写函数体内的语句,便于子类根据自身需要定义自己的行为。Animal b = new Dog();
  • 接口
  • 抽象类、抽象方法

重写规则:

finalstatic方法不可被重写

参数列表:与被重写方法的参数列表必须完全相同

返回类型:与被重写方法的返回类型可以不相同,但是必须是父类返回值的派生类

访问权限:不能比父类中被重写的方法的访问权限更低(public > protected > private > )

抛异常:如果父类方法抛异常,子类不能抛更广泛的异常

同包:除了private final可以重写所有父类方法

不同包:只能重写public 或者 protected的非final方法

子类中调用父类被重写方法可以用super.method()

三、Java中多态的底层实现

多态性特征的最基本体现有“重载”和“重写”,其实这两个体现在Java虚拟机中时分派的作用。分派又分为静态分派和动态分派,静态分派是指所有依赖静态类型来定位方法执行版本的分派动作,动态分派是指在运行期根据实际类型确定方法执行版本的分派过程。

 Animal animal = new Bird();

在上面代码中Animal是父类,Bird是继承Animal的子类;那么在定义animal对象时前面的“Animal”称为变量的静态类型(Static Type),或者叫外观类型(Apparent Type),后面的“Bird”则称为变量的实际类型(Actual Type),静态类型和实际类型在程序中都可以发生一些变化,区别是静态类型的变化仅仅在使用时发生,变量本身的静态类型不会被改变,并且最终的静态类型是在编译器可知的;而实际类型变化的结果在运行期才可以确定,编译器在编译程序的时候并不知道一个对象的实际对象是什么。

//实际类型变化
Animal bird = new Bird();
Animal eagle = new Eagle();
//静态类型变化
sd.sayHello((Bird)bird);
sd.sayHello((Eagle)eagle);

animal对象的静态类型是Animal,实际类型是Bird。静态类型在编译期可知,实际类型在运行时可知。

四、重载

同一个类中相同方法名的不同方法。

重载,使用哪个重载版本,就完全取决于传入参数的数量数据类型
方法重载是通过静态分派实现的,静态分派是发生在编译阶段,因此匹配到静态类型Animal。

例如下面代码:

package test;

/**
* @Description: 方法静态分派演示
* @version: v1.0.0
 */
public class StaticDispatch {

	static abstract class Animal{ }

	static class Bird extends Animal{ }

	static class Eagle extends Animal{ }

	public void sayHello(Animal animal) {
		System.out.println("hello,animal");
	}

	public void sayHello(Bird bird) {
		System.out.println("hello,I'm bird");
	}

	public void sayHello(Eagle eagle) {
		System.out.println("hello,I'm eagle");
	}

	public static void main(String[] args){
	    Animal bird = new Bird(); // 静态类型Animal(编译期可知)--实际类型Bird(运行期可知)
	    Animal eagle = new Eagle(); // 静态类型Animal(编译期可知)--实际类型Eagle(运行期可知)
	    StaticDispatch sd = new StaticDispatch();
	    sd.sayHello(bird);
	    sd.sayHello(eagle);
	}
}
/* 结果:
hello,animal
hello,animal
*/

代码中刻意地定义了两个静态类型相同Animal,实际类型不同的变量,但虚拟机(准确的是编译器)在重载时是通过参数的静态类型而不是实际类型来作为判断依据的。并且静态类型是编译期可知的,因此,在编译阶段,Javac编译器会根据参数的静态类型决定使用哪个重载版本,因此选择了sayHello(Animal)作为调用目标。

方法重载是通过静态分派实现的,并且静态分派是发生在编译阶段,所以确定静态分派的动作实际上不是由虚拟机来执行的;另外,编译器虽然能确定出方法重载的版本,但在很多情况下这个版本并不是“唯一的”,往往只能确定一个“更加适合”的版本。

五、重写

方法的重写:与虚拟机中动态分派的过程有着密切联系。

package test;

/**
* @Description: 方法动态分派演示
* @version: v1.0.0
 */
public class DynamicDispatch {

	static abstract class Animal{
		protected abstract void sayHello();
	}

	static class Bird extends Animal{

		@Override
		protected void sayHello() {
			System.out.println("Bird say hello");
		}

	}

	static class Eagle extends Animal{
		@Override
		protected void sayHello() {
			System.out.println("Eagle say hello");
		}

	}

	public static void main(String[] args){
	    Animal bird = new Bird();
	    Animal eagle = new Eagle();
	    bird.sayHello();
	    eagle.sayHello();
	    bird = new Eagle();
	    bird.sayHello();
	}
}

/* 结果:
Bird say hello
Eagle say hello
Eagle say hello
*/

通过javap反编译命令来看一段该代码的字节码:

>javap -c DynamicDispatch.class
Compiled from "DynamicDispatch.java"
public class com.carmall.DynamicDispatch {
  public com.carmall.DynamicDispatch();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: new           #2                  // class com/carmall/DynamicDispatch$Bird

       3: dup
       4: invokespecial #3                  // Method com/carmall/DynamicDispatch$Bir
d."<init>":()V
       7: astore_1
       8: new           #4                  // class com/carmall/DynamicDispatch$Eagl
e
      11: dup
      12: invokespecial #5                  // Method com/carmall/DynamicDispatch$Eag
le."<init>":()V
      15: astore_2
      16: aload_1
      17: invokevirtual #6                  // Method com/carmall/DynamicDispatch$Ani
mal.sayHello:()V
      20: aload_2
      21: invokevirtual #6                  // Method com/carmall/DynamicDispatch$Animal.sayHello:()V
      24: new           #4                  // class com/carmall/DynamicDispatch$Eagle
      27: dup
      28: invokespecial #5                  // Method com/carmall/DynamicDispatch$Eagle."<init>":()V
      31: astore_1
      32: aload_1
      33: invokevirtual #6                  // Method com/carmall/DynamicDispatch$Animal.sayHello:()V
      36: return
}

上面的指令,invokevirtual表示运行时按照对象的类来调用实例方法;invokespecial根据编译时类型来调用实例方法,也就是会调用父类。

0~15行的字节码是准备动作,作用时建立birdeagle的内存空间、调用BirdEagle类型的实例构造器,将这两个实例的引用存放在第1、2个局部变量表Slot之中。

接下来的16~21行时关键部分;这部分把刚刚创建的两个对象的引用压到栈顶,这两个对象是将要执行的sayHello()方法的所有者,称为接收者(Receiver);17和21句是方法调用命令,这两条调用命令单从字节码角度来看,无论是指令(invokevirtual)还是参数(都是常量池中第6项的常量,注释显示了这个常量是sayHello方法的符号引用)完全一样的,但是这两句执行的目标方法并不同,这是因为invokevirtual指令的多态查找过程引起的,该指令运行时的解析过程可分为以下几个步骤:

  • 找到操作数栈第一个元素所指向的对象的实际类型,记为C。如果在类型C中找到了与常量中描述符和简单名称都一样的方法,则进行访问权限校验,如果通过则返回该方法的的直接引用,查找过程结束;如果不通过,则返回java.lang.IllegalAccessError异常。
  • 否则,按照继承关系从下往上一次对C的各个父类进行第二步的搜索和验证过程。
  • 如果始终都没有找到合适的方法,则抛出java.lang.AbstractMethodError异常。
  • 由于invokevirtual指令执行的第一步就是在运行期确定接收者的实际类型,所以两次调用的invokevirtual指令把常量池中的类方法符号引用解析到了不同的直接引用上,这个过程就是Java语言中重写的本质。
package java.lang;

import java.lang.annotation.*;

/**
 * Indicates that a method declaration is intended to override a
 * method declaration in a supertype. If a method is annotated with
 * this annotation type compilers are required to generate an error
 * message unless at least one of the following conditions hold:
 *
 * <ul><li>
 * The method does override or implement a method declared in a
 * supertype.
 * </li><li>
 * The method has a signature that is override-equivalent to that of
 * any public method declared in {@linkplain Object}.
 * </li></ul>
 *
 * @author  Peter von der Ah&eacute;
 * @author  Joshua Bloch
 * @jls 9.6.1.4 @Override
 * @since 1.5
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}

到此这篇关于Java基础之面向对象机制(多态、继承)底层实现的文章就介绍到这了,更多相关Java面向对象机制底层实现内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • JAVA序列化和反序列化的底层实现原理解析

    一.基本概念 1.什么是序列化和反序列化 (1)Java序列化是指把Java对象转换为字节序列的过程,而Java反序列化是指把字节序列恢复为Java对象的过程: (2)**序列化:**对象序列化的最主要的用处就是在传递和保存对象的时候,保证对象的完整性和可传递性.序列化是把对象转换成有序字节流,以便在网络上传输或者保存在本地文件中.序列化后的字节流保存了Java对象的状态以及相关的描述信息.序列化机制的核心作用就是对象状态的保存与重建. (3)**反序列化:**客户端从文件中或网络上获得序列化后

  • java中面向对象的概念及知识点总结

    作为java中的一个重要理念,说起面向对象也是老生常谈了.在找资料的时候多是很专业的术语,又或者很多框架的知识点合集,其实大部分人刚看资料的时候是不能理解的.本篇对面向对象的一些基础概念进行介绍,尽量以通俗的语言展现给大家,分为概念.好处.与面向过程区别和最重要的三大特征. 1.概念 面向过程是以函数为基础,完成各种操作,强调过程:就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步实现,使用的时候一个一个依次调用就可以了. 2.设计和开发程序的好处 (1)交流更加流畅 (2)提高设计和

  • 深入学习Java同步机制中的底层实现

    前言 在多线程编程中我们会遇到很多需要使用线程同步机制去解决的并发问题,而这些同步机制就是多线程编程中影响正确性和运行效率的重中之重.这不禁让我感到好奇,这些同步机制是如何实现的呢?好奇心是进步的源泉,就让我们一起来揭开同步机制源码的神秘面纱吧. 在本文中,我们会从JDK中大多数同步机制的共同基础AbstractQueuedSynchronizer类开始说起,然后通过源码了解我们最常用的两个同步类可重入锁ReentrantLock和闭锁CountDownLatch的具体实现.通过这篇文章我们将可

  • Java CAS底层实现原理实例详解

    这篇文章主要介绍了Java CAS底层实现原理实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 一.CAS(compareAndSwap)的概念 CAS,全称Compare And Swap(比较与交换),解决多线程并行情况下使用锁造成性能损耗的一种机制. CAS(V, A, B),V为内存地址.A为预期原值,B为新值.如果内存地址的值与预期原值相匹配,那么将该位置值更新为新值.否则,说明已经被其他线程更新,处理器不做任何操作:无论哪种情

  • Java ArrayList的底层实现方法

    如下所示: package com.soto.collection; /** * 自己实现一个ArrayList,帮助我们更好地理解ArrayList的底层结构; * @author 王 * */ public class SxtArrayList { private Object[] elementData; private int size; public int size(){ return size; } public boolean isEmpty(){ return size ==

  • Java synchronize底层实现原理及优化

    首先来说下synchronize和Lock的区别: 两者都是锁,用来控制并发冲突,区别在于Lock是个接口,提供的功能更加丰富,除了这个外,他们还有如下区别: synchronize自动释放锁,而Lock必须手动释放,并且代码中出现异常会导致unlock代码不执行,所以Lock一般在Finally中释放,而synchronize释放锁是由JVM自动执行的. Lock有共享锁的概念,所以可以设置读写锁提高效率,synchronize不能.(两者都可重入) Lock可以让线程在获取锁的过程中响应中断

  • Java并发底层实现原理学习心得

    我们知道java实现的并发操作最后肯定是由我们的CPU完成的,中间经历了将java源码编译成.class文件,然后进行加载,然后虚拟机执行引擎进行执行,解释为汇编语言,然后转为操作系统指令,然后转为1,0,最后CPU进行识别执行. 提到java的并发,我们不由的就会想到java中常见的键字:volatile和synchronized,我们接下来就会从这两个关机字展开分析: volatile的底层实现原理 synchronized的实现原理和应用 volatile 说到volatile,在java

  • 浅谈Java 并发的底层实现

    并发编程的目的是让程序运行更快,但是使用并发并不定会使得程序运行更快,只有当程序的并发数量达到一定的量级的时候才能体现并发编程的优势.所以谈并发编程在高并发量的时候才有意义.虽然目前还没有开发过高并发量的程序,但是学习并发是为了更好理解一些分布式架构.那么当程序的并发量不高,比如是单线程的程序,单线程的执行效率反而比多线程更高.这又是为什么呢?熟悉操作系统的应该知道,CPU是通过给每个线程分配时间片的方式实现多线程的.这样,当CPU从一个任务切换到另一个任务的时候,会保存上一个任务的状态,当执行

  • Java基础详解之面向对象的那些事儿

    一.面向对象的理解 首先,要清楚一点,与面向过程不同,面向对象有着自己独特的优势.举个通俗的例子,设想我们要得到一把椅子,面向过程的话我们可能需要通过买木材,设计图纸,最后切割而成.但对于面向对象来说,我们只需找到一个家具店,买一把椅子即可. 面向对象也是Java语言的特点之一,其实从idea中不难看出,这种面向对象的语言所具有: 1.易维护 2.易复用 3.易扩展 二.三大特性 2.1 继承(extends): 顾名思义,是获取原有的基础,在此基础上进行操作的一个过程.使用已存在的类的定义作为

  • Java基础之面向对象机制(多态、继承)底层实现

    一.Java的前世 为什么会产生Java?Java的特点是什么? 从C语言开始讲,C语言是一种结构化语言,模块化编程,便于程序的调试,依靠非常全面的运算符和多样的数据类型,可以轻易完成各种数据结构的构建,通过指针类型更可对内存直接寻址以及对硬件进行直接操作,因此既能够用于开发系统程序,也可用于开发应用软件.其缺点就是封装性弱,程序的安全性上不是很好.C语言的异常处理一般使用setjmp()与longjmp(),在捕获到异常时进行跳转:或者使用abort()和exit()两个函数,强行终止程序的运

  • Java全面分析面向对象之多态

    目录 多态的理解 向上转型 向上转型的三种形式 动态绑定和静态绑定 方法的重写 进一步认识和理解多态 多态的优点 多态的理解 什么是多态呢??从字面理解就是多种形态,也就是不同类实例化出来的对象调用同一种方法,也可以理解为不同类的对象经过同一种行为产生的状态是不同的,这就是多态. 要想理解多态,我们必须要了解向上转型和重写这两个重点然后在来深刻理解多态这一概念,等看完向上转型与重写再来看多态的概念,你就会豁然开朗,一下就通透了不少.因为多态的条件就是向上转型,重写以及继承. 向上转型 首先多态的

  • 详解Java基础篇--面向对象1(构造方法,static、this关键字)

    面向对象,面向过程的区别.拿下五子棋来说: 面向过程分析: 开始游戏 黑棋先走 绘制画面 判断输赢 轮到白棋 绘制画面 判断输赢 返回步骤2 输出结果 面向对象分析: 黑白双方,双方行为是一模一样的 棋盘系统,负责绘制画面 规则系统,判断犯规.输赢 传统的面向过程编程是思考问题的解决步骤,这种思维方式适用于问题规模较小时.可是当问题规模大,要求程序有更好的可扩展性,能更快速地查错时面向对象设计思想就能体现出其优势.面向对象更接近人类地自然思维方式,将现实世界中的事物抽象为对象和对象的方法. 面向

  • Java基础教程之接口的继承与抽象类

    在实施接口中,我们利用interface语法,将interface从类定义中独立出来,构成一个主体.interface为类提供了接口规范. 在继承中,我们为了提高程序的可复用性,引入的继承机制.当时的继承是基于类的.interface接口同样可以继承,以拓展原interface. 接口继承 接口继承(inheritance)与类继承很类似,就是以被继承的interface为基础,增添新增的接口方法原型.比如,我们以Cup作为原interface: 复制代码 代码如下: interface Cup

  • Java轻松掌握面向对象的三大特性封装与继承和多态

    目录 1.封装 1.介绍 2.封装的理解和好处 3.封装的实现步骤 2.继承 1.介绍 2.继承的基本语法 3.继承的使用细节 3.super关键字 1.基本介绍 2.基本语法 3.细节与好处 4.super与this的比较 4.方法重写 1.基本介绍 2.注意事项与使用细节 3.重载与重写的比较 3.多态 1.基本介绍 2.具体体现 1.方法的多态 2.对象的多态(重点) 3.多态注意事项和细节讨论 1.多态的前提 2.属性 3.instance of 4.多态的向上转型 5.多态的向下转型

  • Java面向对象程序设计:继承,多态用法实例分析

    本文实例讲述了Java面向对象程序设计:继承,多态用法.分享给大家供大家参考,具体如下: 本文内容: 继承 多态 首发时期:2018-03-23 继承: 介绍: 如果多个类中存在相同的属性和行为,可以将这些内容抽取到单独一个类中,那么多个类(子类)无需再定义这些属性和行为,只要继承那个类(父类/超类/基类)即可.[比如,泰迪狗类也是狗类,狗的属性,泰迪狗也应该是有的,那么泰迪狗类在定义的时候就不必要添加那些与狗类重复的属性了,而是直接继承狗类即可.在多个狗类的子类(泰迪狗,二哈,柴犬)的时候,直

  • Java基础之方法重写和多态示例

    本文实例讲述了Java基础之方法重写和多态.分享给大家供大家参考,具体如下: 示例代码: 子类的方法与父类的方法具有相同的返回类型和签名,子类方法重写父类方法. class A{ int i,j; //父类变量 A(int a,int b){ //父类构造 i=a; j=b; } void show(){ //父类方法 System.out.println(i+" "+j); } } class B extends A{ //继承父类 int k; //子类变量 B(int a,int

  • Java基础之ArrayList的扩容机制

    我们知道Java中的ArrayList对象底层是基于数组实现的,而数组是有长度限制的,那基于数组实现的ArrayList是否有长度限制呢?我们通过ArrayList的构造方法来剖析 ArrayList提供了3种构造方法以便我们来获取: ArrayList(int initialCapacity) 第一种需要赋值长度进行new ArrayList() 第二种无参构造,不需要赋值数组初始长度 ArrayList(Collection<? extends E> c) 第三种入参一个继承了Collec

  • Java 基础语法之解析 Java 的包和继承

    目录 一.包 1. 概念 2. 使用方式 3. 静态导入 4. 创建包 5. 包的访问权限 6. 常见的系统包 二.继承 1. 概念 2. 语法规则(含 super 使用) 3. protected 关键字 4. 更复杂的继承关系 5. final 关键字 三.组合 四.总结(含谜底) 一.包 1. 概念 根据定义:包是组织类的一种方式 那么为什么要组织类呢? 简单来讲就是保证类的唯一性,就比如在以后的工作中,如果大家一起开发一个项目,大家可能在自己的代码中都写到了一个 Test 类,而如果出现

随机推荐