第二章 内部类(最难的)
2.1 概述
2.1.1 什么是内部类
将一个类A定义在另一个类B里面,里面的那个类A就称为内部类,B则称为外部类。可以把内部类理解成寄生,外部类理解成宿主。
2.1.2 什么时候使用内部类
一个事物内部还有一个独立的事物,内部的事物脱离外部的事物无法独立使用
人里面有一颗心脏。
汽车内部有一个发动机。
为了实现更好的封装性。
2.2 内部类的分类
按定义的位置来分
-
静态内部类,类定义在了成员位置 (类中方法外称为成员位置,有static修饰的内部类)(了解)
-
成员内部内,类定义在了成员位置 (类中方法外称为成员位置,无static修饰的内部类)(了解)
-
局部内部类,类定义在方法内(了解)
-
匿名内部类。一般定义在方法中,或者可执行代码中(掌握)
2.3 静态内部类(了解)
要求掌握如下几点:

两个小问题:

静态内部类特点:
有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();
}
}
总结:

2.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
}
}
}
总结:

2.6 局部内部类(仅限了解,完全不用掌握)
-
局部内部类 :定义在方法中的类。
定义格式:
class 外部类名 {
数据类型 变量名;
修饰符 返回值类型 方法名(参数列表) {
// …
class 内部类 {
// 成员变量
// 成员方法
}
}
}

2.7 匿名内部类【重点!掌握!】
2.7.1 概述
什么是匿名?匿名就是没有名字的意思。
匿名内部类 :是内部类的简化写法。它的本质是一个带具体实现的 父类或者父接口的 匿名的 子类对象。 开发中,最常用到的内部类就是匿名内部类了。
2.7.2 引入
实际上,如果我们希望定义一个只要使用一次的类,就可考虑使用匿名内部类。匿名内部类的本质作用**:是为了方便创建子类对象,最终是为了简化代码**。
之前我們使用接口时,似乎得做如下几步操作:
-
定义子类
-
重写接口中的方法
-
创建子类对象
-
调用重写后的方法
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 匿名内部类的特点
定义一个没有名字的内部类
这个类实现了父类,或者父类接口
匿名内部类会创建这个没有名字的类的对象
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();
}
}
总结:

强调一下上图最后一点,如果匿名内部类 是 new 接口{
},则后面的匿名内部类和该接口是实现关系。
如果是匿名内部类是 new 类{
},则创建的匿名内部类对象和该类是继承关系。也就是该类的子类



















