基本介绍
一个类的内部又完整的嵌套了另一个类结构。被嵌套的类称为内部类(inner class),嵌套其他类的类称为外部类(outer class)。是我们类的第五大成员【属性、方法、构造器、代码块、内部类】。内部类最大的特点就是可以直接访问私有属性,并且可以体现类与类之间的包含关系。注意:内部类是学习的难点,同时也是重点,后面看底层源码时,有大量的内部类。
基本语法
class Outer { //外部类
class Inner { //内部类
}
}
class Other{
}
class Outer { //外部类
private int n1 = 100; //属性
public Outer(int n1) { //构造器
this.n1 = n1;
}
public void m1() { //方法
System.out.println("m1()");
}
{ //代码块
System.out.println("代码块...");
}
class Inner { //内部类,在Outer类的内部
}
}
内部类的分类
a。定义在外部类局部位置上(比如方法内):
(1)局部内部类(有类名)
(2)匿名内部类(没有类名,重点)
b。定义在外部类的成员位置上:
(1)成员内部类(没用static修饰)
(2)静态内部类(使用static修饰)
局部内部类
局部内部类的使用
说明:局部内部类是定义在外部类的局部位置,比如方法中,并且有类名
(1)可以直接访问外部类的所有成员,包含私有的
(2)不能添加访问修饰符,因为它的地位就是一个局部变量。局部变量是不能使用修饰符的。但是可以使用final修饰,因为局部变量也可以使用final
(3)作用域:仅仅在定义它的方法或代码块中
(4)局部内部类----访问---->外部类的成员【访问方式:直接访问】
(5)外部类----访问---->局部内部类的成员
访问方式:创建对象,再访问(注意:必须在作用域内)
//main方法 //外部其他类
Outer02 outer02 = new Outer02();
outer02.m1();
//验证解读
System.out.println("outer02 的 hashcode = " + outer02);
//========================================================
class Outer02 { //外部类
private int n1 = 100;
private void m2() { System.out.println("Outer02 m2()"); } //私有方法
public void m1() { //方法
//1.局部内部类是定义在外部类的局部位置,通常在方法
//3.不能添加访问修饰符,但是可以使用final修饰
//4.作用域:仅仅在定义它的方法或代码块中
String name = "xxx";
final class Inner02 { //局部内部类(本质仍然是一个类)
//2.可以直接访问外部类的所有成员,包含私有的
public int n1 = 800;
public void f1() {
//5.局部内部类可以直接访问外部类的成员,比如下面 外部类n1 和m2()
//7.如果外部类和局部内部类的成员重名时,默认遵循就近原则,如果想访问外部类的成员,
// 则可以使用【外部类名.this.成员】去访问
// 老韩解读:Outer02.this本质就是外部类的对象,即哪个对象调用了m1,Outer02.this就是哪个对象
System.out.println("n1 = " + n1 + " 外部类的n1 = " + Outer02.this.n1);
//验证解读
System.out.println("Outer02.this hashcode = " + Outer02.this);
m2();
}
}
//6.外部类在方法中,可以创建Inner02对象,然后调用方法即可
Inner02 inner02 = new Inner02();
inner02.f1();
}
}
记住:
(1)局部内部类定义在方法中/代码块
(2)作用域在方法体或者代码块中
(3)本质仍然是一个类
(6)外部其他类----不能访问---->局部内部类(因为 局部内部类地位是一个局部变量)
(7)如果外部类和局部内部类的成员重名时,默认遵循就近原则,如果想访问外部类的成员,则可以使用 (外部类名.this.成员)去访问
匿名内部类
匿名内部类的使用(重要!!!)
//(1)本质是类 (2)内部类 (3)该类没有名字 (4)同时还是一个对象
说明:匿名内部类是定义在外部类的局部位置,比如方法中,并且没有类名
匿名内部类的基本语法
new 类或接口 (参数列表) {
类体
};
/**
* 演示匿名内部类的使用
*/
public class AnnoymousInnerClass {
public static void main(String[] args) {
Outer04 outer04 = new Outer04();
outer04.method();
}
}
class Outer04 { // 外部类
private int n1 = 10; //属性
public void method() { //方法
//基于接口的匿名内部类
//解读
//1.需求:想使用IA接口,并创建对象
//2.传统方式,是写一个类,实现该接口,并创建对象 例如:Tiger/Dog
//3.现需求 Tiger/Dog 类只是使用一次,后面不再使用
//4.可以使用匿名内部类来简化开发
//5.tiger的编译类型? IA
//6.tiger的运行类型? 就是匿名内部类 XXXX => Outer04$1
/*
我们看底层 会分配 类名 Outer04$1
class XXXX implements IA {
@Override
public void cry() {
System.out.println("老虎叫唤...");
}
}
*/
//7.jdk底层在创建匿名内部类 Outer04$1 ,立即马上就创建了 Outer04$1实例,并且
// 把地址返回给了tiger
//8.匿名内部类使用一次,就不能再使用
IA tiger = new IA() {
@Override
public void cry() {
System.out.println("老虎叫唤...");
}
};
System.out.println("tiger的运行类型 = " + tiger.getclass());
tiger.cry();
/*
IA tiger = new Tiger();
tiger.cry();
*/
//演示基于类的匿名内部类
//分析
//1.father编译类型 Father
//2.father运行类型 Outer04$2
//3.底层会创建匿名内部类
/*
class Outer04$2 extends Father {
@Override
public void test() {
System.out.println("匿名内部类重写了test方法");
}
}
*/
//4.同时也直接返回了 匿名内部类 Outer04$2的对象
//5.注意 ("jack") 参数列表会传递给 构造器
Father father = new Father("jack") {
@Override
public void test() {
System.out.println("匿名内部类重写了test方法");
}
};
System.out.println("father对象的运行类型 = " + father.getClass());
father.test();
//基于抽象类的匿名内部类
Animal animal = new Animal() {
@Override
void eat() {
System.out.println("吃东西...");
}
};
animal.eat();
}
}
interface IA { //接口
public void cry();
}
/*
class Tiger implements IA {
@Override
public void cry() {
System.out.println("老虎叫唤...");
}
}
class Dog implements IA {
@Override
public void cry() {
System.out.println("小狗叫唤...");
}
}
*/
class Father {
public Father(String name) { //构造器
super();
}
public void test() { //方法
}
}
abstract class Animal { //抽象类
abstract void eat();
}
匿名内部类的使用细节
(2)匿名内部类的语法比较奇特,请大家注意,因为匿名内部类既是一个类的定义,同时它本身也是一个对象,因此从语法上看,它既有定义类的特征,也有创建对象的特征,对前面代码分析可以看出这个特点,因此可以调用匿名内部类方法。
(3)可以直接访问外部类的所有成员,包含私有的
(4)不能添加访问修饰符,因为他的地位就是一个局部变量
(5)作用域:仅仅在定义它的方法或代码块中
(6)匿名内部类----访问---->外部类成员【访问方式:直接访问】
(7)外部其他类----不能访问---->匿名内部类(因为匿名内部类地位是一个局部变量)
(8)如果外部类和匿名内部类的成员重名时,匿名内部类访问的话,默认遵循就近原则,如果想访问外部类的成员,则可以使用(外部类名.this.成员)去访问
//main方法
Outer05 outer05 = new Outer05();
outer05.f1();
System.out.println("main outer05 hashcode = " + outer05);
class Outer05 {
private int n1 = 99;
public void f1() {
//创建一个基于类的匿名内部类
//不能添加访问修饰符,因为他的地位就是一个局部变量
//作用域:仅仅在定义它的方法或代码块中
Person p = new Person() {
private int n1 = 88;
@Override
public void hi() {
//可以直接访问外部类的所有成员,包含私有的
//如果外部类和匿名内部类的成员重名时,匿名内部类访问的话,
//默认遵循就近原则,如果想访问外部类的成员,
//则可以使用(外部类名.this.成员)去访问
System.out.println("匿名内部类重写了 hi方法 n1 = " + n1
+ "外部类的n1 = " + Outer05.this.n1);
//Outer05.this就是调用f1的对象
System.out.println("Outer05.this hashcode = " + Outer05.this);
}
};
p.hi(); //动态绑定,运行类型是Outer05$1
//也可以直接调用,匿名内部类本身也是返回对象
// class 匿名内部类 extends Person {}
new Person() {
@Override
public void hi() {
System.out.println("匿名内部类重写了 hi方法, 哈哈哈...");
}
@Override
public void ok(String str) {
super.ok(str);
}
}.ok("jack");
}
}
class Person { // 类
public void hi() {
System.out.println("Person hi()...");
}
public void ok(String str) {
System.out.println("Person ok()..." + str);
}
}
匿名内部类的最佳实践
当做实参直接传递,简洁高效。
public class InnerClassExercise01 {
public static void main(String[] args) {
//当做实参直接传递,简洁高效
f1(new IL() {
@Override
public void show() {
System.out.println("这是一幅名画...");
}
});
//传统方法
f1(new Picture());
}
//静态方法,形参是接口类型
public static void f1(IL il) {
il.show();
}
}
//接口
interface IL {
void show();
}
//类->实现IL接口 =>编程领域(硬编码)
class Picture implements IL {
@Override
public void show() {
System.out.println("这是一幅名画...");
}
}
课堂练习
1.有一个铃声接口Bell,里面有个ring方法。
2.有一个手机类Cellphone,具有闹钟功能alarmclock,参数是Bell类型
3.测试手机类的闹钟功能,通过匿名内部类(对象)作为参数,打印:懒虫起床了
4.再传入另一个匿名内部类(对象),打印:小伙伴上课了
public class InnerClassExercise02 {
public static void main(String[] args) {
Cellphone cellphone = new Cellphone();
//1.传递的是实现了Bell接口的匿名内部类 InnerClassExercise02$1
//2.重写了ring
//3. Bell bell = new Bell() {
// @Override
// public void ring() {
// System.out.println("懒猪起床了");
// }
// }
cellphone.alarmClock(new Bell() {
@Override
public void ring() {
System.out.println("懒猪起床了");
}
});
cellphone.alarmClock(new Bell() {
@Override
public void ring() {
System.out.println("小伙伴上课了");
}
});
}
}
interface Bell { //接口
void ring(); //方法
}
class Cellphone { //类
public void alarmClock(Bell bell) { //形参是Bell接口类型
bell.ring(); //动态绑定
}
}
匿名内部类:涉及(1)继承(2)多态(3)动态绑定(4)内部类
成员内部类
说明:成员内部类是定义在外部类的成员位置,并且没有static修饰
(1)可以直接访问外部类的所有成员,包含私有的
(2)可以添加任意访问修饰符(public/protected/默认/private),因为它的地位就是一个成员。
(3)作用域:和外部类的其他成员一样,为整个类体,比如前面案例,在外部类的成员方法中创建成员内部类对象,再调用方法
(4)成员内部类----访问---->外部类成员(比如:属性)【访问方式:直接访问】
(5)外部类----访问---->成员内部类【访问方式:创建对象,再访问】
(6)外部其他类----访问---->成员内部类
(7)如果外部类和内部类的成员重名时,内部类访问的话,默认遵循就近原则,如果想访问外部类的成员,则可以使用【外部类名.this.成员】去访问
静态内部类
静态内部类的使用
说明:静态内部类是定义在外部类的成员位置,并且有static修饰
(1)可以直接访问外部类的所有静态成员,包含私有的,但不能直接访问非静态成员
(2)可以添加任意访问修饰符(public、protected、默认、private),因为它的地位就是一个成员
(3)作用域:同其他的成员,为整个类体
(4)静态内部类----访问---->外部类(比如静态属性)【访问方式:直接访问所有静态成员】
(5)外部类----访问---->静态内部类【访问方式:创建对象,再访问】
(6)外部其他类----访问---->静态内部类
(7)如果外部类和静态内部类的成员重名时,静态内部类访问时,默认遵循就近原则,如果想访问外部类的成员,则可以使用【外部类名.成员】去访问
小结
(1)内部类有四种:局部内部类、匿名内部类、成员内部类、静态内部类
(2)重点掌握匿名内部类的使用
new 类/接口 (参数列表){
};
(3)成员内部类、静态内部类是放在外部类的成员位置,本质就是一个成员
(4)其他细节看笔记