java 内部类(inner class)详解

一、为何使用内部类

  • 内部类提供了更好的封装,只有外部类能访问内部类
  • 内部类可以独立继承一个接口,不受外部类是否继承接口影响
  • 内部类中的属性和方法即使是外部类也不能直接访问,相反内部类可以直接访问外部类的属性和方法,即使private
  • 利于回调函数的编写

一个内部类的例子

public class OuterClass {
    private String outerName;
    private int outerAge;
    public class InnerClass{
        private String innerName;
        private int innerAge;
    }
}

二、内部类与外部类的联系

2.1 内部类是一个相对独立的实体,与外部类不是is-a关系

内部类是一个编译时概念,编译后外部类及其内部类会生成两个独立的class文件: OuterClass.classOuterClass$InnerClass.class,我用javac编译器对上面的OuterClass进行编译:

 D:\>javac OuterClass.class

编译后的结果:

这里写图片描述

2.2 内部类可以直接访问外部类的元素,但是外部类不可以直接访问内部类的元素

public class OuterClass {

    private String outerName;
    private int outerAge;

    public class InnerClass{
        private int innerName;
        InnerClass(){
            //内部类可以访问外部类的元素
            outerName="I am outer class";
            outerAge=23;
        }
        public void display(){
            System.out.println(outerName+" and my age is "+outerAge);
        }
    }
    public static void main(String[] args) {
        OuterClass outerClass = new OuterClass();
        OuterClass.InnerClass innerClass = outerClass.new InnerClass();
        innerClass.display();
    }
}

在上面例子中我们可以看到,内部类可以直接访问外部类属性,尽管外部类属性是用private修饰的。这是因为在创建外部类时,内部类会自动捕获一个外部类的引用,所以内部类访问外部类元素,实际上是通过他所持有外部类引用访问的。在java中,我们可以通过OuterClass.this来获得外部类的引用,请看下面例子:

public class OuterClass {
    public void display(){
        System.out.println("this is OuterClass...");
    }
    public class InnerClass{
        //获取外部类的引用
        public OuterClass getOuterClass(){
            return OuterClass.this;
        }
        public void innerDisplay(){
            //内部类也可以通过外部类的引用访问外部元素
            getOuterClass().display();
        }
    }
    public static void main(String[] args) {
        OuterClass outerClass = new OuterClass();
        OuterClass.InnerClass innerClass = outerClass.new InnerClass();
        innerClass.innerDisplay();
    }
}

2.3 外部类可以通过内部类引用间接访问内部类元素

public class OuterClass {
    public void display(){
        //外部类访问内部类元素,需要通过内部类引用访问
        InnerClass innerClass=new InnerClass();
        innerClass.innerDisplay();
    }
    public class InnerClass{
        public void innerDisplay(){
            System.out.println("I am inner class");
        }
    }
    public static void main(String[] args) {
        OuterClass outerClass=new OuterClass();
        outerClass.display();
    }
}

三、创建内部类

3.1 在外部类外面(或外部类main方法)创建内部了对象

其实上面2.2例子中我们已经看到了如何创建内部类。如果要创建一个内部类对象,必须利用outerClass.new来创建:

OuterClass outerClass = new OuterClass();
OuterClass.InnerClass innerClass = outerClass.new InnerClass();

其实我们还可以一步到位:

OuterClass.InnerClass innerClass=new OuterClass().new InnerClass();

内部类创建方法示例:

public static void main(String[] args) {
    //先创建外部类对象,再创建内部类对象
    OuterClass outerClass = new OuterClass();
    OuterClass.InnerClass innerClass1 = outerClass.new InnerClass();
    innerClass1.innerDisplay();
    //一步到位创建
    OuterClass.InnerClass innerClass2=new OuterClass().new InnerClass();
    innerClass2.innerDisplay();
}

3.2 在外部类里面创建内部类

正如2.3代码中display()方法那样,在外部类里面创建内部类,就像创建普通对象一样直接创建:

InnerClass innerClass=new InnerClass()

四、内部类的种类:

在Java中内部类主要分为成员内部类、方法内部类、匿名内部类、静态内部类。

4.1 成员内部类

成员内部类也是最普通的内部类,它是外部类的一个成员,所以他是可以无限制的访问外围类的所有成员属性和方法,尽管是private的,但是外围类要访问内部类的成员属性和方法则需要通过内部类实例来访问。

在成员内部类中要注意两点:

  1. 成员内部类中不能存在任何static的变量和方法
  2. 成员内部类是依附于外围类的,所以只有先创建了外围类才能够创建内部类

4.2 方法内部类

方法内部类定义在外部类的方法中,局部内部类和成员内部类基本一致,只是它们的作用域不同,方法内部类只能在该方法中被使用,出了该方法就会失效。 对于这个类的使用主要是应用与解决比较复杂的问题,想创建一个类来辅助我们的解决方案,到那时又不希望这个类是公共可用的,所以就产生了局部内部类。

4.3 匿名内部类

匿名内部类其实就是一个没有名字的方法内部类,所以它符合方法内部类的所有约束,初次之外,还有一些地方需要注意:

  1. 匿名内部类是没有访问修饰符的。
  2. 匿名内部类必须继承一个抽象类或者实现一个接口
  3. 匿名内部类中不能存在任何静态成员或方法
  4. 匿名内部类是没有构造方法的,因为它没有类名。

一般使用匿名内部类的场景是,要继承或实现的接口只有一个抽象方法,比如添加一个监听器:

public class Button {
    public void click(){
        //匿名内部类,实现的是ActionListener接口
        new ActionListener(){
            public void onAction(){
                System.out.println("click action...");
            }
        }.onAction();
    }
    //匿名内部类必须继承或实现一个已有的接口
    public interface ActionListener{
        public void onAction();
    }

    public static void main(String[] args) {
        Button button=new Button();
        button.click();
    }
}

4.4 静态内部类

关键字static可以修饰成员变量、方法、代码块,其实它还可以修饰内部类,使用static修饰的内部类我们称之为静态内部类。静态内部类与非静态内部类之间存在一个最大的区别,我们知道非静态内部类在编译完成之后会隐含地保存着一个引用,该引用是指向创建它的外围内,但是静态内部类却没有。没有这个引用就意味着:

  1. 静态内部类的创建是不需要依赖于外围类,可以直接创建
  2. 静态内部类不可以使用任何外围类的非static成员变量和方法,而内部类则都可以

    public class OuterClass {
        private static String outerName;
        public  int age;
    
        static class InnerClass1{
            /* 在静态内部类中可以存在静态成员 */
            public static String _innerName = "static variable";
            public void display(){
                /*
                 * 静态内部类只能访问外部类的静态成员变量和方法
                 * 不能访问外部类的非静态成员变量和方法
                 */
                System.out.println("OutClass name :" + outerName);
            }
        }
        class InnerClass2{
            /* 非静态内部类中不能存在静态成员 */
            public String _innerName = "no static variable";
            /* 非静态内部类中可以调用外部类的任何成员,不管是静态的还是非静态的 */
            public void display(){
                System.out.println("OuterClass name:" + outerName);
                System.out.println("OuterClass age:" + age);
            }
        }
        public void display(){
            /* 外部类能直接访问静态内部类静态元素 */
            System.out.println(InnerClass1._innerName);
            /* 静态内部类可以直接创建实例不需要依赖于外部类 */
            new InnerClass1().display();
            /* 非静态内部的创建需要依赖于外部类 */
            OuterClass.InnerClass2 inner2 = new OuterClass().new InnerClass2();
            /* 非静态内部类的成员需要使用非静态内部类的实例访问 */
            System.out.println(inner2._innerName);
            inner2.display();
        }
    
        public static void main(String[] args) {
            OuterClass outer = new OuterClass();
            outer.display();
        }
    }
    
  • 52
    点赞
  • 114
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: Java内部类是指定义在另一个类内部的类,它可以访问该类的所有成员,包括私有的。一个内部类可以定义为static或者非static,用来访问外部类的成员变量和方法,实现代码复用和访问控制。例如,可以使用内部类来定义一个类的内部构造函数,或者定义一个类的内部枚举。 ### 回答2: Java内部类是指在一个类的内部定义另一个类。它的存在可以提供更好的封装性和组织性。 举例来说,假设有一个外部类叫做OuterClass,它包含一个成员变量和一个方法。在OuterClass内部,我们可以定义一个内部类InnerClass。这里是一个具体的例子: ```java public class OuterClass { private int outerVariable; public OuterClass(int value) { outerVariable = value; } public void outerMethod() { InnerClass innerObject = new InnerClass(); innerObject.innerMethod(); } public class InnerClass { public void innerMethod() { System.out.println("外部变量的值是:" + outerVariable); } } } ``` 在这个例子中,OuterClass有一个构造函数用于初始化外部变量outerVariable,还有一个outerMethod方法。内部类InnerClass定义在OuterClass内部,并且可以访问外部变量outerVariable。 当我们创建一个OuterClass的实例时,可以调用outerMethod方法。在outerMethod方法内部,我们创建了InnerClass的一个实例innerObject,并调用它的innerMethod方法。innerMethod方法输出了外部变量outerVariable的值。 通过内部类,我们可以更好地封装和组织相关的代码。内部类可以访问外部类的成员变量和方法,甚至可以使用private修饰的成员。这种封装性和组织性可以帮助我们实现更清晰和复用性更强的代码。 ### 回答3: Java内部类是指在一个类的内部定义的另一个类。与常规类不同的是,内部类具有访问外部类的成员和方法的能力。 举个例子,假设我们有一个外部类叫做"OuterClass",它包含一个名为"outerMethod()"的方法。我们想在"outerMethod()"方法中访问一个局部变量,同时我们希望这个局部变量也能被内部类中的方法使用。这时我们可以在"outerMethod()"方法内部定义一个内部类,让这个内部类可以访问外部类的局部变量。 以下是具体的代码示例: ```java public class OuterClass { private int outerVariable = 10; // 外部类的成员变量 public void outerMethod() { int localVariable = 20; // 外部方法的局部变量 // 定义一个内部类 class InnerClass { public void innerMethod() { System.out.println("访问外部类成员变量: " + outerVariable); System.out.println("访问外部方法局部变量: " + localVariable); } } // 创建内部类对象 InnerClass inner = new InnerClass(); inner.innerMethod(); // 调用内部类的方法 } public static void main(String[] args) { OuterClass outer = new OuterClass(); outer.outerMethod(); } } ``` 在上面的例子中,我们在"OuterClass"类的"outerMethod()"方法中定义了一个内部类"InnerClass"。在"InnerClass"中,我们可以直接访问外部类的成员变量"outerVariable"和外部方法的局部变量"localVariable"。在"outerMethod()"方法中,我们创建了"InnerClass"的实例"inner",并调用了内部类的"innerMethod()"方法。 通过这个例子,我们可以看到Java内部类的强大之处,它能够方便地访问外部类的成员和方法,同时也提供了更好的代码封装和组织方式。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值