zoukankan      html  css  js  c++  java
  • Java内部类

    内部类的基本结构                                                                    

    //外部类
    class Out {
        private int age = 12;
         
        //内部类
        class In {
            public void print() {
                System.out.println(age);
            }
        }
    }
     
    public class Demo {
        public static void main(String[] args) {
            Out.In in = new Out().new In();
            in.print();
            //或者采用下种方式访问
            /*
            Out out = new Out();
            Out.In in = out.new In();
            in.print();
            */
        }
    }

    运行结果:12

    从上面的例子不难看出,内部类其实严重破坏了良好的代码结构,但为什么还要使用内部类呢?

    因为内部类可以随意使用外部类的成员变量(包括私有)而不用生成外部类的对象,这也是内部类的唯一优点如同心脏可以直接访问身体的血液,而不是通过医生来抽血

    程序编译过后会产生两个.class文件,分别是Out.class和Out$In.class

    其中$代表了上面程序中Out.In中的那个 .

    Out.In in = new Out().new In()可以用来生成内部类的对象,这种方法存在两个小知识点需要注意

      1.开头的Out是为了标明需要生成的内部类对象在哪个外部类当中

      2.必须先有外部类的对象才能生成内部类的对象,因为内部类的作用就是为了访问外部类中的成员变量

    内部类中的变量访问形式                                                             

    class Out {
        private int age = 12;
         
        class In {
            private int age = 13;
            public void print() {
                int age = 14;
                System.out.println("局部变量:" + age);
                System.out.println("内部类变量:" + this.age);
                System.out.println("外部类变量:" + Out.this.age);
            }
        }
    }
     
    public class Demo {
        public static void main(String[] args) {
            Out.In in = new Out().new In();
            in.print();
        }
    }

    运行结果:

    局部变量:14
    内部类变量:13
    外部类变量:12

    内部类在没有同名成员变量和局部变量的情况下,内部类会直接访问外部类的成员变量,而无需指定Out.this.属性名

    否则,内部类中的局部变量会覆盖外部类的成员变量

    而访问内部类本身的成员变量可用this.属性名,访问外部类的成员变量需要使用Out.this.属性名

    静态内部类                                                                            

    class Out {
        private static int age = 12;
         
        static class In {
            public void print() {
                System.out.println(age);
            }
        }
    }
     
    public class Demo {
        public static void main(String[] args) {
            Out.In in = new Out.In();
            in.print();
        }
    }

    运行结果:12

    可以看到,如果用static 将内部内静态化,那么内部类就只能访问外部类的静态成员变量,具有局限性

    其次,因为内部类被静态化,因此Out.In可以当做一个整体看,可以直接new 出内部类的对象(通过类名访问static,生不生成外部类对象都没关系)

    私有内部类                                                                            

    class Out {
        private int age = 12;
         
        private class In {
            public void print() {
                System.out.println(age);
            }
        }
        public void outPrint() {
            new In().print();
        }
    }
     
    public class Demo {
        public static void main(String[] args) {
            //此方法无效
            /*
            Out.In in = new Out().new In();
            in.print();
            */
            Out out = new Out();
            out.outPrint();
        }
    }

    运行结果:12

    如果一个内部类只希望被外部类中的方法操作,那么可以使用private声明内部类

    上面的代码中,我们必须在Out类里面生成In类的对象进行操作,而无法再使用Out.In in = new Out().new In() 生成内部类的对象

    也就是说,此时的内部类只有外部类可控制

    如同是,我的心脏只能由我的身体控制,其他人无法直接访问它

    方法内部类                                                                            

    class Out {
        private int age = 12;
     
        public void Print(final int x) {
            class In {
                public void inPrint() {
                    System.out.println(x);
                    System.out.println(age);
                }
            }
            new In().inPrint();
        }
    }
     
    public class Demo {
        public static void main(String[] args) {
            Out out = new Out();
            out.Print(3);
        }
    }

    运行结果:

    3
    12

    在上面的代码中,我们将内部类移到了外部类的方法中,然后在外部类的方法中再生成一个内部类对象去调用内部类方法

    如果此时我们需要往外部类的方法中传入参数,那么外部类的方法形参必须使用final定义

    至于final在这里并没有特殊含义,只是一种表示形式而已

    匿名内部类                                                                            

    public class Goods3 {  
        public Contents cont() {  
            return new Contents() {  
                private int i = 11;  
                public int value() {  
                    return i;  
                }  
            };  
        }  
    }

    java的匿名内部类的语法规则看上去有些古怪,不过如同匿名数组一样,当你只需要创建一个类的对象而且用不上它的名字时,使用内部类可以使代码看上去简洁清楚。它的语法规则是这样的:

    new interfacename(){......}; 或 new superclassname(){......};

    这里方法cont()使用匿名内部类直接返回了一个实现了接口Contents的类的对象,看上去的确十分简洁。

    有一点需要注意的是,匿名内部类由于没有名字,所以它没有构造函数(但是如果这个匿名内部类继承了一个只含有带参数构造函数的父类,创建它的时候必须带上这些参数,并在实现的过程中使用super关键字调用相应的内容)。如果你想要初始化它的成员变量,有下面几种方法:

    如果是在一个方法的匿名内部类,可以利用这个方法传进你想要的参数,不过记住,这些参数必须被声明为final。

    将匿名内部类改造成有名字的局部内部类,这样它就可以拥有构造函数了。

    在这个匿名内部类中使用初始化代码块。

    我是天王盖地虎的分割线                                                             

    参考:http://blog.csdn.net/ilibaba/article/details/3866537

    参考:http://www.cnblogs.com/nerxious/archive/2013/01/24/2875649.html

  • 相关阅读:
    手把手教你利用create-nuxt-app脚手架创建NuxtJS应用
    初识NuxtJS
    webpack打包Vue应用程序流程
    用选择器代替表格列的筛选功能
    Element-UI
    Spectral Bounds for Sparse PCA: Exact and Greedy Algorithms[贪婪算法选特征]
    Sparse Principal Component Analysis via Rotation and Truncation
    Generalized Power Method for Sparse Principal Component Analysis
    Sparse Principal Component Analysis via Regularized Low Rank Matrix Approximation(Adjusted Variance)
    Truncated Power Method for Sparse Eigenvalue Problems
  • 原文地址:https://www.cnblogs.com/yydcdut/p/4000356.html
Copyright © 2011-2022 走看看