第二章 内部类(最难的)
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 类{
},则创建的匿名内部类对象和该类是继承关系。也就是该类的子类