java多态的理解及深入认识

来源:互联网 发布:查看本地数据库ip 编辑:程序博客网 时间:2024/05/16 12:26

           最近开始深入学习java,对于java的一些基本内容还不太理解,在学习过程中遇到了很多问题,其中对于java 封装,继承,多态的问题,网上查阅了很多资料,这里分享一下自己的理解。(刚入门,很多知识还属于入门阶段,)

     1.封装

          封装的对象是java的类,其作用是将java类的内部的数据和具体实现细节隐藏,仅仅提供给外界其访问接口,起到了保护类的作用。

     2.继承

         继承是指从已存在的类中间派生出新的类,新的类,即子类能够继承已存在的类,即父类中的属性和方法,并且能够重写和重载这些方法,为多态的实现做铺垫。

     3.多态

         多态可以说是java三大特性中最重要的一个特性,多态的实现基础是继承,只有存在父类和子类的情况下,才有多态的实现。
        这里总结关于多态的3个便要条件:
         1.必须要有继承
         2..在子类中要重写在父类中已存在的方法(是重写而不是重载,注意区分两者的区别)。
         3.父类的引用变量要指向子类的对象实例。
         多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。因为在程序运行时才确定具体的类,这样,不用修改源程序代码,就可以让引用变量绑定到各种不同的类实现上,从而导致该引用调用的具体方法随之改变,即不修改程序代码就可以改变程序运行时所绑定的具体代码,让程序可以选择多个运行状态,这就是多态性。


         这里引用一篇讲解多态实例的文章,对于理解多态分厂有帮助:


          如果你是一个酒神,对酒情有独钟。某日回家发现桌上有几个杯子里面都装了白酒,从外面看我们是不可能知道这是些什么酒,只有喝了之后才能够猜出来是何种酒。你一喝,这是剑南春、再喝这是五粮液、再喝这是酒鬼酒….在这里我们可以描述成如下:

      酒 a = 剑南春

      酒 b = 五粮液

      酒 c = 酒鬼酒

      …

      这里所表现的的就是多态。剑南春、五粮液、酒鬼酒都是酒的子类,我们只是通过酒这一个父类就能够引用不同的子类,这就是多态——我们只有在运行的时候才会知道引用变量所指向的具体实例对象。

      诚然,要理解多态我们就必须要明白什么是“向上转型”。在继承中我们简单介绍了向上转型,这里就在啰嗦下:在上面的喝酒例子中,酒(Win)是父类,剑南春(JNC)、五粮液(WLY)、酒鬼酒(JGJ)是子类。我们定义如下代码:

      JNC a = new  JNC();

      对于这个代码我们非常容易理解无非就是实例化了一个剑南春的对象嘛!但是这样呢?

      Wine a = new JNC();

      在这里我们这样理解,这里定义了一个Wine 类型的a,它指向JNC对象实例。由于JNC是继承与Wine,所以JNC可以自动向上转型为Wine,所以a是可以指向JNC实例对象的。这样做存在一个非常大的好处,在继承中我们知道子类是父类的扩展,它可以提供比父类更加强大的功能,如果我们定义了一个指向子类的父类引用类型,那么它除了能够引用父类的共性外,还可以使用子类强大的功能。

      但是向上转型存在一些缺憾,那就是它必定会导致一些方法和属性的丢失,而导致我们不能够获取它们。所以父类类型的引用可以调用父类中定义的所有属性和方法,对于只存在与子类中的方法和属性它就望尘莫及了---1。

  
public class Wine {    public void fun1(){        System.out.println("Wine 的Fun.....");        fun2();    }        public void fun2(){        System.out.println("Wine 的Fun2...");    }}public class JNC extends Wine{    /**     * @desc 子类重载父类方法     *        父类中不存在该方法,向上转型后,父类是不能引用该方法的     * @param a     * @return void     */    public void fun1(String a){        System.out.println("JNC 的 Fun1...");        fun2();    }        /**     * 子类重写父类方法     * 指向子类的父类引用调用fun2时,必定是调用该方法     */    public void fun2(){        System.out.println("JNC 的Fun2...");    }}public class Test {    public static void main(String[] args) {        Wine a = new JNC();        a.fun1();    }}-------------------------------------------------Output:Wine 的Fun.....JNC 的Fun2...

从程序的运行结果中我们发现,a.fun1()首先是运行父类Wine中的fun1().然后再运行子类JNC中的fun2()。

      分析:在这个程序中子类JNC重载了父类Wine的方法fun1(),重写fun2(),而且重载后的fun1(String a)与 fun1()不是同一个方法,由于父类中没有该方法,向上转型后会丢失该方法,所以执行JNC的Wine类型引用是不能引用fun1(String a)方法。而子类JNC重写了fun2() ,那么指向JNC的Wine引用会调用JNC中fun2()方法。

      所以对于多态我们可以总结如下:

      指向子类的父类引用由于向上转型了,它只能访问父类中拥有的方法和属性,而对于子类中存在而父类中不存在的方法,该引用是不能使用的,尽管是重载该方法。若子类重写了父类中的某些方法,在调用该些方法的时候,必定是使用子类中定义的这些方法(动态连接、动态调用)。

      对于面向对象而已,多态分为编译时多态和运行时多态。其中编辑时多态是静态的,主要是指方法的重载,它是根据参数列表的不同来区分不同的函数,通过编辑之后会变成两个不同的函数,在运行时谈不上多态。而运行时多态是动态的,它是通过动态绑定来实现的,也就是我们所说的多态性。


 参考资料:http://www.cnblogs.com/chenssy/p/3372798.html

0 0
原创粉丝点击