JavaSE学习进阶day03_02 内部类
第二章 内部类(最难的)
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 = "黑马程序"; // 内部类: Inner01public 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); // 10System.out.println("心脏在跳动 " + this.weight); // 20System.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. 重写抽象方法@Overridepublic 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() {@Overridepublic void swimming() {System.out.println("自由泳...");}}.swimming(); // 接口 变量 = new 实现类(); // 多态,走子类的重写方法Swim s2 = new Swim() {@Overridepublic void swimming() {System.out.println("蛙泳...");}}; s2.swimming();s2.swimming();} }
深入理解匿名内部类:
new Swim() {@Overridepublic void swimming() {System.out.println("蛙泳...");} };
1、首先你需要掌握一点,匿名内部类是哪一个部分?如下:
{@Overridepublic void swimming() {System.out.println("蛙泳...");} };
这是内部类的部分,只是这个类是没有名字的。
2、其次理解 new Swim( )是什么意思。它不是创建一个接口,Swim是接口,new是创建实现这个接口的对象。而这个类是没有名字的。因此整段代码的意思是创建一个实现匿名内部类的对象,且这个对象实现了Swim( )接口。因此匿名内部类里面必须得实现所有接口的抽象方法。
3、掌握如下代码的含义
Swim s2 = new Swim() {@Overridepublic 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() {@Overridepublic 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 类{
},则创建的匿名内部类对象和该类是继承关系。也就是该类的子类