姓名: 李小娜
[嵌牛导读]:封装、继承、多态三大特征是java中比较常用的,务必要掌握,下面给大家介绍Java封装、继承、多态三大特征的理解,有不清楚的朋友可以一起学习下
[嵌牛鼻子]:封装 java的继承 java中的多态
[嵌牛提问]:有几种多态机制?
[嵌牛正文] :首先先简单的说一下其3大特性的定义:
封装:隐藏对象的属性和实现细节,仅对外公开接口,控制在程序中属性的读和修改的访问级别。将抽象得到的数据和行为(或功能)相结合,形成一个有机的整体,也就是将数据与操作数据的源代码进行有机的结合,形成“类”,其中数据和函数都是类的成员。封装的目的是增强安全性和简化编程,使用者不必了解具体的实现细节,而只是要通过外部接口,一特定的访问权限来使用类的成员。封装的基本要求是: 把所有的属性私有化,对每个属性提供getter和setter方法,如果有一个带参的构造函数的话,那一定要写一个不带参的构造函数。在开发的时候经常要对已经编写的类进行测试,所以在有的时候还有重写toString方法,但这不是必须的。
继承:通过继承实现代码复用。Java中所有的类都是通过直接或间接地继承java.lang.Object类得到的。继承而得到的类称为子类,被继承的类称为父类。子类不能继承父类中访问权限为private的成员变量和方法。子类可以重写父类的方法,及命名与父类同名的成员变量。但Java不支持多重继承,即一个类从多个超类派生的能力。在开发中尽量减少继承关系,这样做是为了把程序的耦合度降低。
多态:多态又分为设计时多态和运行时多态,例如重载又被称为设计时多态,而对于覆盖或继承的方法,JAVA运行时系统根据调用该方法的实例的类型来决定选择调用哪个方法则被称为运行时多态。总而言之,面向对象的设计的典型特点就是继承,封装和多态,这些特点也是面向对象之所以能如此盛行的关键所在。
封装
java中类的属性的访问权限默认值不是private,要想隐藏该属性的方法,就可以加private修饰符,来限制只能够在类的内部进行访问。
对于类中的私有属性,要对其给出一对方法(getXXX,setXXX())访问私有属性,保证对私有属性的操作和安全性。
方法的封装,该公开的公开,该隐藏的隐藏。
java的继承
继承,是对有着共同特性的多类事物,进行再抽象成一个类。
java中的继承要使用extends关键字,并且java中指允许单继承,也就是一个类只能有一个父类。
构造方法不能被继承。
java方法中的覆盖
子类中有和父类中可访问的同名同返回同参数列表的方法时,就会覆盖从父类继承来的方法。
super()关键字
super(),表示在子类的构造方法调用父类的构造方法时,super()也只能在构造方法中的第一句。
java中的多态
有两种多态的机制:编译时多态、运行时多态
1、方法的重载:重载是指同一类中有多个同名的方法,但这些方法有着不同的参数。,因此在编译时就可以确定到底调用哪个方法,它是一种编译时多态。
2、方法的覆盖:子类可以覆盖父类的方法,因此同样的方法会在父类中与子类中有着不同的表现形式。在java语言中,基类的引用变量不仅可以指向基类的实例对象,也可以指向子类的实例对象,同样,接口中的引用变量也可以指向其实现类的实例对象。
publicclassA {
publicString show(D obj) {
return("A and D");
}
publicString show(A obj) {
return("A and A");
}
}
publicclassBextendsA{
publicString show(B obj){
return("B and B");
}
publicString show(A obj){
return("B and A");
}
}
publicclassCextendsB{
}
publicclassDextendsB{
}
publicclassTest {
publicstaticvoidmain(String[] args) {
A a1 =newA();
A a2 =newB();
B b =newB();
C c =newC();
D d =newD();
System.out.println("1--"+ a1.show(b));
System.out.println("2--"+ a1.show(c));
System.out.println("3--"+ a1.show(d));
System.out.println("4--"+ a2.show(b));
System.out.println("5--"+ a2.show(c));
System.out.println("6--"+ a2.show(d));
System.out.println("7--"+ b.show(b));
System.out.println("8--"+ b.show(c));
System.out.println("9--"+ b.show(d));
}
}
1--A and A
2--A and A
3--A and D
4--B and A
5--B and A
6--A and D
7--B and B
8--B and B
9--A and D
当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。
这我们用一个例子来说明这句话所代表的含义:a2.show(b);
这里a2是引用变量,为A类型,它引用的是B对象,因此按照上面那句话的意思是说有B来决定调用谁的方法,所以a2.show(b)应该要调用B中的show(B
obj),产生的结果应该是“B and
B”,但是为什么会与前面的运行结果产生差异呢?这里我们忽略了后面那句话“但是这儿被调用的方法必须是在超类中定义过的”,那么show(B
obj)在A类中存在吗?根本就不存在!所以这句话在这里不适用?那么难道是这句话错误了?非也!其实这句话还隐含这这句话:它仍然要按照继承链中调用方法的优先级来确认。所以它才会在A类中找到show(A
obj),同时由于B重写了该方法所以才会调用B类中的方法,否则就会调用A类中的方法。