JavaSE学习进阶day03_02 内部类

第二章 内部类(最难的)

2.1 概述

2.1.1 什么是内部类

将一个类A定义在另一个类B里面,里面的那个类A就称为内部类,B则称为外部类。可以把内部类理解成寄生,外部类理解成宿主。

2.1.2 什么时候使用内部类

一个事物内部还有一个独立的事物,内部的事物脱离外部的事物无法独立使用

  1. 人里面有一颗心脏。

  2. 汽车内部有一个发动机。

  3. 为了实现更好的封装性。

2.2 内部类的分类

按定义的位置来分

  1. 静态内部类,类定义在了成员位置 (类中方法外称为成员位置,有static修饰的内部类)(了解)

  2. 成员内部内,类定义在了成员位置 (类中方法外称为成员位置,无static修饰的内部类)(了解)

  3. 局部内部类,类定义在方法内(了解)

  4. 匿名内部类。一般定义在方法中,或者可执行代码中(掌握)

2.3 静态内部类(了解)

要求掌握如下几点:

JavaSE学习进阶day03_02 内部类_第1张图片

两个小问题:

JavaSE学习进阶day03_02 内部类_第2张图片

静态内部类特点

  • 有static修饰的内部类,属于外部类本身的。

  • 总结:静态内部类与其他类的用法完全一样。只是访问的时候需要加上:外部类.内部类。

  • 拓展:静态内部类可以直接访问外部类的静态成员。

内部类的使用格式

外部类.内部类。

静态内部类对象的创建格式

外部类.内部类  变量 = new  外部类.内部类构造器;

案例演示

// 外部类:Outer01
class Outer01{
​
    private static  String sc_name = "黑马程序";
​
    // 内部类: Inner01
    public static class Inner01{
        // 这里面的东西与类是完全一样的。
        private String name;
​
        public Inner01(String name) {
            this.name = name;
        }
​
        public void showName(){
            System.out.println(this.name);
            // 拓展:静态内部类可以直接访问外部类的静态成员。
            System.out.println(sc_name);
        }
    }
}
​
public class InnerClassDemo01 {
    public static void main(String[] args) {
        // 创建静态内部类对象。
        // 外部类.内部类  变量 = new  外部类.内部类构造器;
        Outer01.Inner01 in  = new Outer01.Inner01("张三");
        in.showName();
    }
}

总结:

JavaSE学习进阶day03_02 内部类_第3张图片

2.4 成员内部类(了解)

掌握以下几点:

JavaSE学习进阶day03_02 内部类_第4张图片

和静态内部类创建对象时不太一样哦。

成员内部类特点

  • 无static修饰的内部类,属于外部类对象的。

  • 宿主:外部类对象。

内部类的使用格式

 外部类.内部类。 // 访问内部类的类型都是用 外部类.内部类

成员内部类创建对象格式

外部类.内部类 变量 = new 外部类构造器.new 内部类构造器;

案例演示

public class InnerClassDemo02 {
    public static void main(String[] args) {
        //  宿主:外部类对象。
       // Outer02 out = new Outer02();
        // 创建内部类对象。
        Outer02.Inner02 in = new Outer02().new Inner02("张三");
        in.showName();
    }
}
​
class Outer02 {
    // 成员内部类,属于外部类对象的。
    // 拓展:成员内部类不能定义静态成员。
    public class Inner02{
        // 这里面的东西与类是完全一样的。
        private String name;
​
        public Inner02(String name) {
            this.name = name;
        }
​
        public void showName(){
            System.out.println(this.name);
        }
    }
}

2.5 成员内部类面试题

请在?地方向上相应代码,以达到输出的内容

注意:内部类访问外部类对象的格式是:外部类名.this

public class Demo05 {
    public static void main(String[] args) {
        Body.Heart heart = new Body().new Heart();
        heart.jump();
    }
}
​
class Body {    // 身体
    private int weight = 30;
​
    // 在成员位置定义一个类
    class Heart {
        private int weight = 20;
​
        public void jump() {
            int weight = 10;
            System.out.println("心脏在跳动 " + weight);  // 10
            System.out.println("心脏在跳动 " + this.weight); // 20
            System.out.println("心脏在跳动 " + Body.this.weight);    // 30
        }
    }
}

总结:

JavaSE学习进阶day03_02 内部类_第5张图片

2.6 局部内部类(仅限了解,完全不用掌握)

  • 局部内部类 :定义在方法中的类。

定义格式:

class 外部类名 {
    数据类型 变量名;
    
    修饰符 返回值类型 方法名(参数列表) {
        // …
        class 内部类 {
            // 成员变量
            // 成员方法
        }
    }
}

JavaSE学习进阶day03_02 内部类_第6张图片

2.7 匿名内部类【重点!掌握!】

2.7.1 概述

什么是匿名?匿名就是没有名字的意思。

匿名内部类 :是内部类的简化写法。它的本质是一个带具体实现的 父类或者父接口的 匿名的 子类对象。 开发中,最常用到的内部类就是匿名内部类了。

2.7.2 引入

实际上,如果我们希望定义一个只要使用一次的类,就可考虑使用匿名内部类。匿名内部类的本质作用**:是为了方便创建子类对象,最终是为了简化代码**。

之前我們使用接口时,似乎得做如下几步操作:

  1. 定义子类

  2. 重写接口中的方法

  3. 创建子类对象

  4. 调用重写后的方法

interface Swim {
    public abstract void swimming();
}
​
// 1. 定义接口的实现类
class Dog implements Swim {
    // 2. 重写抽象方法
    @Override
    public void swimming() {
        System.out.println("狗刨式...");
    }
}
​
public class Demo07 {
    public static void main(String[] args) {
        // 3. 创建实现类对象
        Dog s = new Dog();
        // 4. 调用方法
        s.swimming();
    }
}

我们的目的,最终只是为了调用方法,那么能不能简化一下,把以上四步合成一步呢?匿名内部类就是做这样的快捷方式。

2.7.3 匿名内部类前提和格式

匿名内部类是没有名字的类。

匿名内部类必须继承一个父类或者实现一个父接口

匿名内部类格式

new 父类名或者接口名(){
    // 方法重写
    @Override 
    public void method() {
        // 执行语句
    }
};

2.7.4 使用方式

以接口为例,匿名内部类的使用,代码如下:

创建匿名内部类,并调用。

interface Swim {
    public abstract void swimming();
}
​
public class Demo07 {
    public static void main(String[] args) {
        // 使用匿名内部类
        new Swim() {
            @Override
            public void swimming() {
                System.out.println("自由泳...");
            }
        }.swimming();
​
        // 接口 变量 = new 实现类(); // 多态,走子类的重写方法
        Swim s2 = new Swim() {
            @Override
            public void swimming() {
                System.out.println("蛙泳...");
            }
        };
​
        s2.swimming();
        s2.swimming();
    }
}

深入理解匿名内部类:

   new Swim() {
    @Override
    public void swimming() {
        System.out.println("蛙泳...");
    }
};

1、首先你需要掌握一点,匿名内部类是哪一个部分?如下:

{
    @Override
    public void swimming() {
        System.out.println("蛙泳...");
    }
};

这是内部类的部分,只是这个类是没有名字的。

2、其次理解 new Swim( )是什么意思。它不是创建一个接口,Swim是接口,new是创建实现这个接口的对象。而这个类是没有名字的。因此整段代码的意思是创建一个实现匿名内部类的对象,且这个对象实现了Swim( )接口。因此匿名内部类里面必须得实现所有接口的抽象方法。

3、掌握如下代码的含义

Swim s2 = new Swim() {
    @Override
    public void swimming() {
        System.out.println("蛙泳...");
    }
};

后面那段代码我们知道是创建了一个没有名字的类的对象,我们前面知道,接口是可以接受它所实现类的对象的。因此可以把这个对象赋值给该接口。回顾:接口本身自己是不能创建接口对象的!

2.7.5 匿名内部类的特点

  1. 定义一个没有名字的内部类

  2. 这个类实现了父类,或者父类接口

  3. 匿名内部类会创建这个没有名字的类的对象

2.7.6 匿名内部类的使用场景

通常在方法的形式参数是接口或者抽象类时,也可以将匿名内部类作为参数传递。代码如下:

interface Swim {
    public abstract void swimming();
}
​
public class Demo07 {
    public static void main(String[] args) {
        // 普通方式传入对象
        // 创建实现类对象
        Student s = new Student();
        
        goSwimming(s);
        // 匿名内部类使用场景:作为方法参数传递
        Swim s3 = new Swim() {
            @Override
            public void swimming() {
                System.out.println("蝶泳...");
            }
        };
        // 传入匿名内部类
        goSwimming(s3);
​
        // 完美方案: 一步到位
        goSwimming(new Swim() {
            public void swimming() {
                System.out.println("大学生, 蛙泳...");
            }
        });
​
        goSwimming(new Swim() {
            public void swimming() {
                System.out.println("小学生, 自由泳...");
            }
        });
    }
​
    // 定义一个方法,模拟请一些人去游泳
    public static void goSwimming(Swim s) {
        s.swimming();
    }
}

总结:

JavaSE学习进阶day03_02 内部类_第7张图片

强调一下上图最后一点,如果匿名内部类 是 new 接口{

},则后面的匿名内部类和该接口是实现关系。

如果是匿名内部类是 new 类{

},则创建的匿名内部类对象和该类是继承关系。也就是该类的子类

你可能感兴趣的:(Java学习,学习,java,jvm)