6. Java基础:内部类


一、概述

  • 将类写在其他类的内部,可以写在其他类的 成员位置局部位置,这时写在其他类内部的类就称为内部类。其他类也称为外部类
  • 什么时候使用内部类?

在描述事物时,若一个事物内部还包含其他可能包含的事物,比如在描述汽车时,汽车中还包含这发动机,这时发动机就可以使用内部类来描述。

class 汽车 { //外部类
    class 发动机 { //内部类
    }
}

二、内部类的特点

  • 内部类提供了更好的封装,只有外部类能访问内部类
  • 内部类可以独立继承一个接口,不受外部类是否继承接口影响
  • 内部类可以 直接访问 外部类的成员,包括私有 private
  • 外部类要访问内部类的成员,必须创建对象
  • 在外部类中,即使内部类中用 private 修饰的成员,也可以在外部类中以 内部类 对象.成员 的方式访问
  • private 修饰内部类,则外部类以外不能访问,只能在外部类访问

三、内部类的分类

  • 内部类分为 成员内部类局部内部类
  • 其次还有 匿名内部类静态内部类
  • 定义内部类时,就是一个正常定义类的过程,同样包含各种修饰符、继承与实现关系等。

四、成员内部类

1. 定义

  • 定义在 外部类中的成员位置。与类中的成员变量相似,可通过 外部类.对象 进行访问
  • 访问方式: 外部类名.内部类名 变量名 = new 外部类名().new 内部类名();

2.例子

class Body {//外部类,身体
     private boolean life= true; //生命状态
     public class Heart { //内部类,心脏
         public void jump() {
             System.out.println("心脏噗通噗通的跳")
                System.out.println("生命状态" + life); //访问外部类成员变量
        }
    }
}

//访问内部类
public static void main(String[] args) {
    //创建内部类对象
    Body.Heart bh = new Body().new Heart();
    //调用内部类中的方法
    bh.jump();
}

3. 成员内部类的同名变量调用:

public class Outer {
    int i  = 1;
    class Inner {
        int i  = 2;
        public void inner(){
            int i = 3;
            System.out.println(Outer.this.i);
        }
    }
}

五、局部内部类

1. 概述

  • 局部内部类,定义在外部类 方法中 的局部位置。与访问方法中的局部变量相似,可通过 调用方法 进行访问。
  • 可以直接访问外部类成员
  • 局部内部类,只能在 方法体中 使用
  • 局部类 不能加访问修饰符,因为它们不是类成员。
  • 访问方式: 在外部类方法中,创建内部类对象,进行访问

2. 例子

//定义类
class Party {//外部类,聚会
    public void puffBall(){// 吹气球方法
        class Ball {// 内部类,气球
              public void puff(){
                System.out.println("气球膨胀了");
              }
        }
        //创建内部类对象,调用puff方法
        new Ball().puff();
    }
}
访问内部类
public static void main(String[] args) {    
    //创建外部类对象
    Party p = new Party();
    //调用外部类中的puffBall方法
    p.puffBall();
}

3. 局部内部类访问局部变量的注意事项:

  • 局部内部类访问局部变量必须用 final 修饰
  • jdk8或者更高版本,从语法上讲,不要求被局部内部类所访问的局部变量,一定要加 final
  • 但是,如果在代码中,没有 final,只要局部内部类访问局部变量,编译器会自动给局部变量加 final

为什么一定要加 final

因为局部变量会随着方法的调用而调用,随着调用完毕而消失。
而局部对象并没有立即从堆内存中消失,还要使用那个变量。所以,为了让数据还能继续被使用,就用 fianl 修饰,这样,在堆内存里面存储的其实是一个 常量值

  • 内存图

局部变量被final修饰的问题


六、匿名内部类

1. 概述

  • 定义:就是一个没有名字的 局部内部类。
  • 作用:匿名内部类是创建某个类型子类对象的快捷方式。

定义了一个内部类(成员位置内部类,局部位置内部类),没有名字,意味着,类不能通过名字来复用。不能复用,并不意味着不能使用,它还是可以被使用的,但是 只能被使用一次(在类定义的时候使用一次)

2. 特点:

  • 匿名内部类是 没有 访问修饰符
  • 匿名内部类必须继承一个 或者实现一个 接口

这里的类可以是 具体类 也可以是 抽象类

  • 匿名内部类中不能存在任何 静态成员方法
  • 匿名内部类是 没有构造方法,因为它没有类名
  • 匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。
  • 匿名内部类不能是抽象的,所以,它必须要实现继承的类或者实现的接口的 所有抽象方法

3. 格式:

new 父类或接口(){
    //进行方法重写
};

4. 匿名内部类如果不定义变量引用,则就是匿名对象

public abstract class Person{
    public abstract void eat();
}

new Person(){
    public void eat() {
        System.out.println(“我吃了”);
    }
}.eat();

5. 匿名内部类的使用:

  • 一般使用匿名内部类的场景是,要继承或实现的接口只有一个抽象方法,比如添加一个监听器:
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();
    }
}
  • 其次,当方法的形式参数,是一个接口(绝大部分情况)的时候,在该方法传递实际参数的时候,实际上传递的是一个 接口的子类对象
  • 传递的这个实际参数,就可以用匿名内部类来代替
public class AnoyInnerDemo2 {
    public static void main(String[] args) {
        //1.利用常规方式来调用input方法
        //A a = new A();
        //input(new A());

        //2\. 利用匿名内部类
        input(new Ordinary() {
            @Override
            public void toDo() {
                System.out.println("inner class toDo");
            }
        } );
    }

    //接收接口类型形式参数的方法
    public static void input(Ordinary o) {
        o.toDo();
    }
}

//接口
interface Ordinary {
    void toDo();
}

class A implements Ordinary {
    @Override
    public void toDo() {
        System.out.println("class A todo");
    }
}

七、静态内部类

1. 概述

  • 使用static修饰的 成员内部类 我们称之为静态内部类
  • 静态内部类的创建是不需要依赖于外部类,可以直接创建

2. 访问特征:

  • 对于 静态内部类 而言,它不能访问外部类中 非静态的 成员变量和成员方法
  • 外部类中 访问,静态内部类,和访问普通成员内部类没有任何区别
  • 外部类的外部 访问静态内部类,由于静态内部类,不依赖于外部类对象 : new 外部类类名.内部类类名()

3.例子:

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();
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值