内部类:
一个类的内部又完整的嵌套了另一个类结构。被嵌套的类成为内部类(inner class),嵌套其他类的类称为外部类(outer class)。是类的第五大成员(1、属性 2、方法 3、构造器 4、代码块 5、内部类)
内部类最大的特点就是可以直接访问私有属性,并且可以体现类与类之间的包含关系。
基本语法:
class Outer{ //外部类
class Inner{ //内部类
}
}
class Other{ //外部其他类
}
入门案例:
public class InnerClass{ //外部其他类
public static void main(String[] args) {
}
}
class Outer{
private int n1 = 100; //属性
public Outer(int n1){ //构造器
this.n1 = n1;
}
public void m1(){ //方法
System.out.print("m1()");
}
{ //代码块
System.out.print("代码块~~");
}
class Inner { //内部类,在Outer类的内部
}
}
内部类的分类:
定义在外部类局部位置上(比如方法内):
-
局部内部类(有类名)
-
匿名内部类(没有类名,重点)
定义在外部的成员位置上:
-
成员内部类(没用static修饰)
-
静态内部类(使用static修饰)
局部内部类的使用:
说明:局部内部类是定义在外部类的局部位置,比如方法中,并且有类名。
- 可以直接访问外部类的所有成员,包括私有的
-
不能添加访问修饰符,因为它的地位就是一个局部变量,局部变量是不能使用修饰符的。但是可以使用final修饰,因为局部变量也可以使用final
-
作用域:仅仅在定义它的方法或代码块中
-
局部内部类–访问—>外部类的成员 [访问方式:直接访问]
-
外部类–访问—>局部内部类的成员
访问方式:创建对象,再访问(注意:必须在作用域内)
-
外部其他类–不能访问—>局部内部类(因为 局部内部类地位是一个局部变量)
-
如果外部类和局部内部类的成员重名时,默认遵循就近原则,如果想访问外部类的成员,则可以使用(外部类名.this.成员) 去访问
如:
System.out.print("外部类的n2=" + 外部类名.this.n2);
演示局部内部类的使用
public static void main(String[] args) {
Outer02 outer02 = new Outer02;
outer02.m1();
}
/*
输出结果:
n1=100
Outer02 m2()
*/
----------------------------------------------------------
class Outer02{ //外部类
private int n1 =100;
private void m2(){ //私有方法
System.out.print("Outer02 m2()");
}
public void m1(){ //方法
//1、局部内部类是定义在外部类的局部位置,通常在方法
//3、不能添加访问修饰符,但是可以使用final修饰
class Inner02{ //4、局部内部类(本质仍然是一个类)
/*
如果定义一个与外部类同名的属性
private int n1 = 800;
此时,将输出
n1=100
Outer02 m2()
f1()方法可以写成
public void f1(){ //局部内部类可以直接访问外部类成员
System.out.print("n1=" + n1 + "外部类的n1" + Outer02.this.n1);
m2();
}
*/
//2、可以直接访问外部类的所有成员,包括私有的
public void f1(){ //局部内部类可以直接访问外部类成员
System.out.print("n1=" + n1);
m2();
}
}
class Inner03 extends Inner02{ //如果final class Inner02 就不能继承了
}
//6、外部类在方法中,可以创建Inner02对象,然后调用方法即可
Inner02 inner02 = new Inner02();
inner02.f1();
}
}
匿名内部类
说明:
匿名内部类是定义在外部类的局部位置,比如方法中,并且没有类名
1、本质是一个类,2、是一个内部类,3、该类没有名字,4、同时还是一个对象
- 匿名内部类的基本语法
new 类或接口(参数列表){
类体
};
举例说明:
/*
演示匿名内部类的使用
*/
public Class AnonymousInnerClass{
public static void main(String[] args) {
Outer04 outer04 = new Outer04();
outer02.method();
}
}
class Outer04{ //外部类
private int n1 = 10;
public void method(){
//基于接口的匿名内部类
//1、需求:想使用接口IA
/*
传统方法:写一个类,实现该接口,并创建对象
但,如果这个对象只用一次,以后再也不用的话 创建这么多对象会很浪费
此时,可以使用匿名内部类来简化开发
*/
--------------------------------------
Tiger tiger = new Toger();
tiger.cry();
--------------------------------------
匿名内部类:
/*
编译类型:IA
运行类型:匿名内部类
底层: 会自动分配类名,Outer04$1,这个类用一次就没有了
class Outer04$1 implements IA{
@Override
public void cry(){
System.out.print("老虎叫唤~~");
}
}
*/
/*
jdk底层在创建匿名内部类 Outher04$1,立马就创建了 Outher04$1实例,
并且把地址返回给tiger
匿名内部类使用一次就不能再使用,但对象可以反使用
匿名内部类创建了一个方法后自动销毁,但他的小孩子对象继承了它的方法本领
*/
IA tiger = new IA(){
@Override
public void cry(){
System.out.print("老虎叫唤~~~~");
}
};
tiger.cry();
--------------------------------------
}
}
interface IA{
public void cry();
}
//实现接口
class Tiger implements IA{
@Override
public void cry(){
System.out.print("老虎叫唤~~");
}
}
class Father{
public Father(String name){
}
public void test(){
}
}
abstract class Animal{ //抽象类
abstract void eat();
}
//演示基于类的匿名内部类
Father father = new Father("jack"){
@Override
public void test(){
System.out.print("匿名内部类重写了test方法");
}
};
/*
class Outer04$2 extends Father{} //这样做了之后就是一个新的Outer04$2类,
不再是Father类了
分析:
father编译类型:Father
运行类型:Outer04$2
如果不带大括号,那么 编译类型和运行类型都是Father
*/
//基于抽象类的匿名内部类
new Animal(){
@Override
void eat(){
System.out.print("小狗吃骨头");
}
}
- 匿名内部类既是一个类的定义,同时它本身也是一个对象,因此从语法上看,它既有定义类的特征,也有创建对象的特征,对前面代码分析可以看出来这个特点,因此可以看出这个特点,因此可以调用匿名内部类方法
public class AnonymousInnerClassDetail{
public static void main(String[] args) {
}
}
class Outer05{
private int n1 = 99;
public void f1(){
//创建一个基于类的匿名内部类
Person p = new Person(){
new Person(){
@Override
public void hi(){
System.out.print("匿名内部类重写了hi方法");
}
}
}
p.hi();//动态绑定
/*
编译类型:Person
运行类型:Outer05$1
*/
//也可以直接调用
new Person(){
@Override
public void hi(){
System.out.print("匿名内部类重写了hi方法,哈哈~!");
}
}.hi(); //匿名内部类本身也是返回对象,所以直接.hi()也是允许的
}
}
class Person{ //类
public void hi(){
System.out.print("Person hi()");
}
}
- 可以直接访问外部类的所有成员,包含私有的
- 不能添加访问修饰符,因为它的地位就是一个局部变量
- 作用域:仅仅在定义它的方法或代码块中
- 匿名内部类–访问—>外部类成员 访问方法:直接访问
- 外部其他类–不能访问—>匿名内部类(因为 局部内部类地位是一个局部变量)
- 如果外部类和匿名内部类的成员重名时,默认遵循就近原则,
- 如果想访问外部类的成员,则可以使用(外部类名.this.成员) 去访问
匿名内部类实践:
当作实参直接传递,简洁高效
public class InnerClassExercise01{
public static void main(String[] args) {
//当作实参直接传递,简洁高效
f1(new IL(){
@Override
public void show(){
System.out.print("这是一幅名画~~~");
}
});
}
}
传统写法:
//1、类->实现IL
class Picture implements IL {
@Override
public void show(){
System.out.print("这是一幅名画~~~");
}
}
//2、
f1(new Picture()); //相当于把实现了接口的Picture的对象传给了f1()
------------------------------------------------------
//静态方法,形参是接口类型
public static void f1(IL il){
il.show();
}
//接口
interface IL{
void show();
}