JAVA内部类的作用

作者: EnjoyAndroid | 来源:发表于2017-11-02 01:13 被阅读85次

    定义

    定义在一个类里面的类就叫内部类。

    作用

    1、内部类可以很好的隐藏实现;
    一般的非内部类,是不允许有 private 与protected权限的,但内部类可以
    2、内部类可以实现多重继承;
    3、内部类拥有外部类的所有访问权限;
    4、可以避免父类和接口同方法名时的覆盖问题。

    解析

    1、内部类可以很好的隐藏实现;

    由于外部类是不允许定义为private或者protected类型的,所以如果我们要隐藏一些我们实现细节,就可以通过内部类来实现,比如支付是比较核心的功能,需要尽可能的隐藏其实现细节,调用者只需要能够使用即可,如下:

    public interface IPay{  
        void pay();  
    }  
      
    public class Payer1 {  
    
        private class PayImpl implements IPay{  
          
            public void pay() {  
              System.out.println("pay ing...");
            }  
        }  
      
        public IPay startPay() {  
            return new PayImpl();  
        }  
    }  
      
    
    class TestPay {
        public static void main(String[] args) {  
            Payer1 pm = new Payer1();  
            IPay p = pm.startPay();  
            p.pay();  
        }  
    }  
    

    上面的PayImpl定义在内部类中,用private修饰符来控制访问权限。在后面的main方法中,直接通过IPay.pay()方法进行操作,外部调用者甚至连该实现类的名字都没有看见,这样就可以尽可能的隐藏实现细节,体现java的封装性。

    2、内部类可以实现多重继承;

    现实世界中是存在多重继承关系的,比如,孩子既继承父亲也继承母亲的基因,父亲会work,母亲会dancing,孩子继承了父母的所有这些基因。在java代码中该如何表示呢?由于在java中一个类只支持单继承,多实现,但是对于这种场景,用接口来定义父亲和母亲似乎不太合理,所以这时候内部类就发挥其作用了:

    public class father {
      void work(){...}
    }
    
    public class mother{
      void dancing(){...}
    }
    
    public class Child {
    
      public static void main(String args[])
       {
           Child child=new Child ();
           child.work();
           child.dancing();
       }
    
    
       public void work(){
        InnerChild1 child1 = new InnerChild1 ();
        child1.work();
        }
    
       public void dancing(){
        InnerChild2 child2 = new InnerChild2 ();
        child2.dancing();
        }
    
    
    
        private class InnerChild1 extends father{
    
            void work() {
              super.work();
            }
        }
    
        private class InnerChild2 extends mother{
    
            void dancing() {
              super.dancing();
            }
        }
    }
    

    这样孩子便继承了父母的优秀基因,可谓才艺双馨。

    3、内部类拥有外部类的所有访问权限

    由于非静态内部类会持有外部类的引用,因此,非静态内部类可以访问外部类的所有属性及方法,这样可以为程序的设计带来极大的灵活性。对第一段代码稍作修改:

    public interface IPay{  
        void pay();  
    }  
      
    public class Payer1 {  
       private double money;
    
        private class PayImpl implements IPay{  
          
            public void pay() {  
              money --;
              System.out.println("pay ing...");
            }  
        }  
      
        public IPay startPay() {  
            return new PayImpl();  
        }  
    }  
      
    
    class TestPay {
        public static void main(String[] args) {  
            Payer1 pm = new Payer1 ();  
            IPay p = pm.startPay();  
            p.pay();  
        }  
    }  
    

    我们在外部类中新增了一个用private修饰的money字段,然后在内部类PayImpl 调用pay方法时,修改了该字段的值,这样可以直接通过内部类来访问外部类。

    4、可以避免父类和接口同方法名时的覆盖问题

    试想一下,如果你的类要继承一个类,还要实现一个接口,可是你发觉你继承的类和接口里面有两个同名的方法该怎么办?你怎么区分它们?你可能会说,我改成不同名的不就完事了吗,多大点事?如果这个父类和接口你都能改的话,确实可以这样做,但是有没有办法在不改的前提下解决这个问题呢,或者说你根本就没有机会改,比如,他们是你引入的第三方SDK呢,你该咋办? 这时候就需要我们实现内部类了:

    public class Payer{  
        void pay(){
         System.out.println("Payer implement...");
      }
    }  
    
    public interface IPay{  
        void pay();  
    }  
    
     //上面部分中Payer类和IPay接口有相同的方法pay(),然后我们有如下实现方法:
    
    public class Payer1 extends Payer  implements IPay{  
      
            @override
            public void pay() {  
              System.out.println("pay ing...");
            }  
        
    }  
    //  想问一下大家pay()这个方法是属于覆盖Payer这里的方法呢?还是IPay这里的方法。我怎么能调到Payer这里的方法?显然这是不好区分的。而我们如果用内部类就很好解决这一问题了。看下面代码
    
    public class Payer1 extends Payer {
    
     private class InnerPayer implements IPay {
          public void pay() {
            Payer1.this.pay();
          }
     }
       IPay startPay() {
          return new InnerPayer ();
      }
    }
    
    class TestPay {
        public static void main(String[] args) {  
            Payer1 pm = new Payer1 ();  
            IPay p = pm.startPay();  
            p.pay();  
        }  
    }  
    

    这样就完美的解决了同名方法覆盖的问题。

    相关文章

      网友评论

        本文标题:JAVA内部类的作用

        本文链接:https://www.haomeiwen.com/subject/scdnpxtx.html