java面向对象三大特性

Java面向对象的特性

一.封装

1.1、概念:

  将类的某些信息隐藏在类内部,不允许外部程序直接访问,而是通过该类提供的方法来实现对隐藏信息的操作和访问。

1.2、好处:

  只能通过规定的方法访问数据。

  隐藏类的实例细节,方便修改和实现。 

1.3、封装的实现步骤

     

 需要注意:对封装的属性不一定要通过get/set方法,其他方法也可以对封装的属性进行操作。当然最好使用get/set方法,比较标准。

1.4、访问修饰符

 

  从表格可以看出从上到下封装性越来越差。

1.5、this关键字

 1.this关键字代表当前对象

  this.属性 操作当前对象的属性

  this.方法 调用当前对象的方法。

 2.当getter和setter函数参数名和成员函数名重合的时候,可以使用this区别。如:

  

1.6、Java 中的内部类

 内部类( Inner Class )就是定义在另外一个类里面的类。与之对应,包含内部类的类被称为外部类。

  内部类的主要作用如下:

  1. 内部类提供了更好的封装,可以把内部类隐藏在外部类之内,不允许同一个包中的其他类访问该类。

  2. 内部类的方法可以直接访问外部类的所有数据,包括私有的数据

  3. 内部类所实现的功能使用外部类同样可以实现,只是有时使用内部类更方便。

二、继承

2.1、继承的概念

  继承是类与类的一种关系,是一种“is  a”的关系。比如“狗”继承“动物”,这里动物类是狗类的父类或者基类,狗类是动物类的子类或者派生类。如下图所示:

   

 

 注:java中的继承是单继承,即一个类只有一个父类。

2.2、继承的好处

 子类拥有父类的所有属性和方法(除了private修饰的属性不能拥有)从而实现了实现代码的复用; 

2.3、语法规则

只要在子类加上extends关键字就可以继承相应的父类:

  

2.4、继承的初始化顺序

 1、初始化父类再初始化子类

 2、先执行初始化对象中属性,再执行构造方法中的初始化。

 基于上面两点,我们就知道实例化一个子类,java程序的执行顺序是:

 父类对象属性初始化---->父类对象构造方法---->子类对象属性初始化--->子类对象构造方法   

 下面有个形象的图:

   

2.5、final关键字

 使用final关键字做标识有“最终的”含义。

  1. final 修饰,则该类不允许被继承。

  2. final 修饰方法,则该方法不允许被覆盖(重写)。

  3. final 修饰属性,则该类的该属性不会进行隐式的初始化,所以 该final 属性的初始化属性必须有值,或在构造方法中赋值(但只能选其一,且必须选其一,因为没有默认值!),且初始化之后就不能改了,只能赋值一次。

  4. final 修饰变量,则该变量的值只能赋一次值,在声明变量的时候才能赋值,即变为常量。

2.6、super关键字

 在对象的内部使用,可以代表父类对象。

  1、访问父类的属性:super.age

  2、访问父类的方法:super.eat()

 super的应用:

 子类的构造过程中必须调用父类的构造方法。这个过程我们使用super关键字隐式调用。

 如果自己用super关键字在子类里调用父类的构造方法,则必须在子类的构造方法中的第一行。

 要注意的是:如果子类构造方法中既没有显式调用父类的构造方法,而父类没有无参的构造方法,则编译出错。

(补充说明:子类构造方法没有通过super显式调用父类的有参构造方法,也没有通过this显式调用自身其他的构造方法,系统默认调用父类法人无参构造方法。【例如:Super()】。子类构造方法通过super显式调用父类的有参构造方法,执行父类相应构造方法,而不执行父类无参构造方法。)

2.7、构造方法

  每个类有默认的无参构造方法,但是如果有其他(带参)构造方法,此时会覆盖掉无参构造方法 

补充:

  一个类文件中允许有多个class类,但是public类只能有一个,类名与类文件相同。

三、多态

3.1、概念:

  同一个引用类型,使用不同的实例,执行不同的操作。

  方法重写是实现多态的基础。

3.2、引用多态   

 父类的引用可以指向本类的对象,可以指向子类的对象;

 首先我们创建一个父类Animal和一个子类Dog,在主函数里如下所示:

  

  注意:我们不能使用一个子类的引用来指向父类的对象,如:

  为了深刻理解引用多态的意义,我们能说“狗是一种动物”,但是不能说“动物是一种狗”,狗和动物是父类和子类的继承关系,它们的从属是不能颠倒的。当父类的引用指向子类的对象时,该对象将只是看成一种特殊的父类(里面有重写的方法和属性),反之,一个子类的引用来指向父类的对象是不可行的!!

3.3、方法多态

  根据上述创建的两个对象:本类对象和子类对象,同样都是父类的引用,当我们指向不同的对象时,它们调用的方法也是多态的。

  创建本类对象时,调用的方法为本类方法;

  创建子类对象时,调用的方法为子类重写的方法或者继承的方法;

  使用多态的时候要注意:如果我们在子类中编写一个独有的方法(没有继承父类的方法),此时就不能通过父类的引用创建的子类对象来调用该方法!!!

  注意: 继承是多态的基础。

3.4、抽象类

 定义:抽象类前使用abstract关键字修饰,则该类为抽象类。

 特点:

   抽象类不允许实例化,因为抽象类中方法未具体化,这是一种不完整的类,所以直接实例化没有意义。

   拥有抽象方法的类必须声明抽象类。

     但是抽象类中不一定非要有抽象方法。

     如果子类没有实现父类的所有抽象方法,子类必须被定义为抽象类。

     没有抽象构造方法,也没有抽象静态方法

    抽象类中可以有非抽象的构造方法,创建子类的实例时可能调用

   使用抽象类要注意以下几点:

 1.抽象类应用场景:

   a.父类的某些方法不确定时,可以用abstract关键字来修饰该方法[抽象方法]。

   b.从多个具有相同特征的类中抽象出一个抽象类,以这个抽象类作为子类的模板,从而避免子类设计的随意性。

    2.抽象类定义抽象方法,只有声明,不需要实现。抽象方法没有方法体以分号结束,抽象方法必须用abstract关键字来修饰。抽象方法在子类中必须被实现,除非子类也是抽象类。

如:

 3.包含抽象方法的类是抽象类。抽象类中可以包含普通的方法,也可以没有抽象方法。如:

  

 4.抽象类不能直接创建,可以定义引用变量来指向子类对象,来实现抽象方法。

  例:    

 1 public abstract class Telephone {
 2      public abstract void call();//抽象方法,方法体以分号结束,只有声明,不需要实现
 3      public void message(){
 4         System.out.println("我是抽象类的普通方法");
 5 }//抽象类中包含普通的方法
 6  }
 7 
 8  public class Phone extends Telephone {
 9      public void call() {//继承抽象类的子类必须重写抽象方法
10            System.out.println("我重写了抽象类的方法");
11      }
12 }

 main函数里: 

1 public class train{
2     public static void main(String[] args){
3         //Telephone t=new Telephone();  抽象类不能直接创建
4         Telephone p=new Phone();  //引用变量指向子类
5         p.call();
6         p.message();
7     }
8 }

    运行结果():

    

父类的引用变量可以调用重写父类后的方法,无法调用子类特有的方法。

如果需要调用子类特有的方法,使用“向下转型”。

 

 向上转型:子类对象转为父类,父类可以是接口。

  公式:Father f = new Son();Father是父类或接口,son是子类。

 向下转型:父类对象转为子类。

  公式:Son s = (Son)f;

在向下转型的过程中,如果没有转换为真实子类类型,会出现类型转换异常。

没有转换为真实子类类型的类型转换异常?

  使用instanceof判断父类的对象引用是不是指向子类的实例,是的话,则可以向下转型。对象的类型必须和instanceof后面的参数所指定的类在继承上有上下级关系。

例如:

If(父类 instanceof 子类){

  子类 子类引用对象 =(子类)父类引用对象

  子类引用对象.子类特定的方法

}

3.5、接口

 1、概念

  接口可以理解为一种特殊的类,由全局常量和公共的抽象方法所组成。也可理解为一个特殊的抽象类,因为它含有抽象方法。

 2.接口定义的基本语法

      [修饰符] [abstract] interface 接口名 [extends父接口1,2....](多继承){

      常量 (public static final)                                          

      抽象方法(public abstract)                                      

      }                                                                                         

 3.使用接口

  一个类可以实现一个或多个接口,实现接口使用implements关键字。java中一个类只能继承一个父类,是不够灵活的,通过实现多个接口可以补充。

  继承父类实现接口的语法为:

     [修饰符] class 类名 extends 父类 implements 接口1,接口2...{

       类体部分

     }

  注意:如果要继承父类,继承父类必须在实现接口之前,即extends关键字必须在implements关键字前

  补充说明:通常我们在命名一个接口时,经常以I开头,用来区分普通的类。如:IPlayGame

  以下我们来补充在上述抽象类中的例子,我们之前已经定义了一个抽象类Telephone和子类Phone,这里我们再创建一个IPlayGame的接口,然后在原来定义的两个类稍作修改,代码如下: 

public interface IPlayGame {

     public void paly();//abstract 关键字可以省略,系统会自动加上

     public String name="游戏名字";//static final关键字可以省略,系统会自动加上
}
public class Phone extends Telephone implements IPlayGame{ public void call() {//继承抽象类的子类必须重写抽象方法 System.out.println("我重写了抽象类的方法"); } @Override public void paly() { System.out.println("我重写了接口的方法"); } } public class train {   public static void main(String[] args) {   IPlayGame i=new Phone();//用接口的引用指向子类的对象   i.paly();//调用接口的方法   System.out.println(i.name);//输出接口的常量   }
}

 运行结果:

   

 4.接口和匿名内部类配合使用

    接口在使用过程中还经常和匿名内部类配合使用。匿名内部类就是没有没名字的内部类,多用于关注实现而不关注实现类的名称。

  语法格式:

 1 Interface i =new interface(){
 2 
 3      Public void method{
 4 
 5          System.out.println(“利用匿名内部类实现接口1”);
 6 
 7       }
 8 
 9  };
10 
11  i.method();

  还有一种写法:(直接把方法的调用写在匿名内部类的最后) 

1 Interface i =new interface(){
2 
3       Public void method{   
4 
5           System.out.println(“利用匿名内部类实现接1”);
6 
7    }
8 
9  }.method();    

Static(静态方法)

  可直接通过类名访问

  静态方法中不能使用this和super

  不能直接访问所属类的实例变量和实例方法

  可直接访问类的静态方法和静态变量

 

 

 

posted @ 2019-03-21 20:36  飘向远方  阅读(2067)  评论(0编辑  收藏  举报