封装
1. 封装的理解
封装从字面上来理解就是包装的意思,专业点就是信息隐藏,是指利用抽象数据类型将数据和基于数据的操作封装在一起,使其构成一个不可分割的独立实体,数据被保护在抽象数据类型的内部,尽可能地隐藏内部的细节,只保留一些对外接口使之与外部发生联系
- 关于封装其实没有什么可以多去学习的
继承
1. 继承的理解
继承是使用已存在的类的定义作为基础建立新类的技术,新类的定义可以增加新的数据或新的功能,也可以用父类的功能,但不能选择性地继承父类。通过使用继承我们能够非常方便地复用以前的代码,能够大大的提高开发的效率。
继承所描述的关系,如果有两个对象A和B,若可以描述为“A是B”,则可以表示A继承B,其中B是被继承者称之为父类或者超类,A是继承者称之为子类或者派生类。
同时在使用继承时需要记住三句话:
1、子类拥有父类非private的属性和方法(父亲的私人财产儿子不能继承吧)
。
2、子类可以拥有自己属性和方法,即子类可以对父类进行扩展(儿子也可以拥有自己的私人空间
)。
3、子类可以用自己的方式实现父类的方法(基因遗传
)。
2. 继承的特性
构造器
对于继承而言,子类会默认调用父类的构造器,但是如果没有默认的父类构造器,子类必须要显示的指定父类的构造器,而且必须是在子类构造器中做的第一件事
protected
关键字
对于protected
而言,它指明该类用户而言,他是private
,但是对于任何继承与此类的子类而言或者其他任何位于同一个包的类而言,他却是可以访问的。
诚然尽管可以使用protected
访问修饰符来限制父类属性和方法的访问权限,但是最好的方式还是将属性保持为private
(我们应当一致保留更改底层实现),通过protected
方法来控制类的继承者的访问权限
向上转型
将子类转换成父类,在继承关系上面是向上移动的,所以一般称之为向上转型。由于向上转型是从一个叫专用类型向较通用类型转换,所以它总是安全的,唯一发生变化的可能就是属性和方法的丢失。这就是为什么编译器在“未曾明确表示转型”活“未曾指定特殊标记”的情况下,仍然允许向上转型的原因。
慎用继承
继承缺点
- 父类变,子类就必须变。
- 继承破坏了封装,对于父类而言,它的实现细节对与子类来说都是透明的。
- 继承是一种强耦合关系。
《Think in Java》中提供了解决办法:问一问自己是否需要从子类向父类进行向上转型。如果必须向上转型,则继承是必要的,但是如果不需要,则应当好好考虑自己是否需要继承。
多态
1. java
多态的理解
- 定义: 指允许不同类的对象对同一消息做出响应,即同一消息可以根据发送对象的不同来采取多种不同的行为方式, 也可以理解为中国人是亚洲人,也是人.
- 实现多态的技术: 动态绑定
- 作用:消除类型之间的耦合关系
- 存在的必要条件: 1. 要有继承,2要有重写,3.父类引用指向子类的对象
- 缺点:不能调用子类中有而父类中没有的方法
2. 动态绑定
1. 静态绑定机制
//被调用的类
class Father{
public static void f1(){
System.out.println("Father— f1()");
}
}
//调用静态方法
public class StaticCall{
public static void main(){
Father.f1(); //调用静态方法
}
}
实现原理
-
java
当中的方法只有final
、static
、private
修饰的的方法和构造方法是静态绑定的 -
final
修饰的方法可以被继承,但是不能被子类重写 所以子类中调用的是父类的final方法,final的好处:1. 防止方法覆盖 2. 关闭java的动态绑定 - 然后在
Father
类所在的方法区中找到f1()方法的直接地址,并将这个直接地址记录到StaticCall
类的常量池索引为13(**13** --- 在常量池中的索引。初始化方法名<clinit>,该方法直接由JVM在特定的时候调用,并非由字节码生成。)
的常量表中。这个过程叫常量池解析 ,以后再次调用Father.f1()
时,将直接找到f1方法的字节码。 - 在编译期就能确定调用哪个方法的方式就是静态绑定
2. 动态绑定
//被调用的父类
class Father{
public void f1(){
System.out.println("father-f1()");
}
public void f1(int i){
System.out.println("father-f1() para-int "+i);
}
}
//被调用的子类
class Son extends Father{
public void f1(){ //覆盖父类的方法
System.out.println("Son-f1()");
}
public void f1(char c){
System.out.println("Son-s1() para-char "+c);
}
}
//调用方法
public class AutoCall{
public static void main(String[] args){
Father father=new Son(); //多态
father.f1(); //打印结果: Son-f1()
}
}
- 方法表 :Jvm加载类的同时,会在方法区位这个类存放很多信息,其中有个数据结构:方法表,: 它是数组的形式记录了当前类及所有的超类的可见方法字节码在内存中的直接地址
- 方法表的特点: 1. 子类方法表继承父类的方法 2. 相同的方法(方法名称和参数相同)在所有类的方发表中的索引相同
- 根据对象(father)的声明类型(Father)还不能够确定调用方法f1的位置,必须根据father在堆中实际创建的对象类型Son来确定f1方法所在的位置。这种在程序运行过程中,通过动态创建的对象的方法表来定位方法的方式,我们叫做 动态绑定机制 。