一、成员变量内部类:
- 成员内部类应用率第:因为只能外部类访问内部类,限制利访问
- 类中套类。外面的类称为外部类,里面的成为内部类
- 内部类通常只服务于外部类(因为内部类充当了外部类的成员)
- 内部类对象只能在外部类中创建
- 成员内部类对外不具备可见性
- 内部类 中的可以直接访问外部类的成员(包括私有的),在内部类中有个隐式的引用指向了创建它的外部类对象——外部类名.this
成员内部类对外不具备可见性:如果在main方法中想要new个Boo,编译错误,外边用不了
public class InnerClassDemo {
public static void main(String[] args) {
Mama m = new Mama();
//Baby b = new Baby(); //编译错误,内部类对外不具备可见性
}
}
class Mama{ //外部类
private String name;
Baby b = new Baby(); //内部类对象通常在外部类中创建
class Baby{ //内部类
void show(){
System.out.println(name);
System.out.println(Mama.this.name); //Mama.this指代它的外部类对象
//要不全省,要不全写,不能这么写 System.out.println(this.name);这个指的是Baby中的name
}
外部类.this指的是外部类的对象,类似于普通方法中有个this指代当前对象
二、匿名变量内部类:
匿名内部类:没有名字的内部类
应用比较高——大大简化代码
1、若想创建一个类的对象(派生类)并且对象只能创建一个,此时该类不必命名。成为匿名内部类
2、匿名内部类中不能修改外面变量的值,因为再此处该变量默认为final的(如果外面的变量发生了改变的话,也是不能进行访问,发生编译错误的.因为默认的是final,但是发生了改变会和定义发生冲突)
public class AnonInnerClassDemo {
public static void main(String[] args) {
//1)创建了Aoo的一个派生类,但是没有名字
//2)为该派生类创建了一个对象,名为o1
//3)大括号中的为派生类的类体
Aoo o1 = new Aoo(){
};
//1)创建了Aoo的一个派生类,但是没有名字
//2)为该派生类创建了一个对象,名为o2
//3)大括号中的为派生类的类体
Aoo o2 = new Aoo(){
};
int num = 5;
num = 55;
//1)创建了Boo的一个派生类,但是没有名字
//2)为该派生类创建了一个对象,名为o3
//3)大括号中的为派生类的类体
Boo o3 = new Boo(){
void show(){
System.out.println("showshow");
//num = 66; //编译错误,匿名内部类中不能修改外面变量的值,因为在此处默认为final的
}
};
o3.show();
}
}
abstract class Boo{
abstract void show();
}
abstract class Aoo{
}
理解:匿名内部类的作用
- 不用去单独去创建派生类,然后再去继承,这里一句话就能实现了
- 匿名内部类是针对一个派生类,且该派生类只创建一次
- 若想创建一个类的对象(派生类)并且对象只创建一个,此时该类不必命名,做成匿名内部类即可
- 若想创建多个派生类的对象,还是需要一个一个创建
Boo o3 = new Eoo();
Class Eoo extends Boo{}
//匿名内部类等同上面的代码
补充:
1、隐式对象:
this:当前对象
super:当前对象的超类对象
外部类名.this:当前对象的外部类对象
2、必须记住的,API中会用到:
1)外部类名.this:当前对象的外部类对象
2)匿名内部类中不能修改外面变量的值,因为在此处该变量默认为final的
3、小面试题
内部类中有独立的.class字节码文件吗?
答:有。内部类也是个类,只要是类进行编译都会产生字节码文件
out文件中放的是字节码文件
4、做功能的套路:
- 先写行为/方法:
1)若为某些对象所特有的行为,就将方法设计在特定的类中
2)若为所有对象所共有的行为,就将方法设计在超类中
先写行为/方法
1)若为派生类所特有的行为,就将方法设计在特定的类中
2)若为派生类所共有的行为
调错方式:
1)打桩:System.out.println(数据);
2)在main方法中把所有的行为都遮住,一条一条松开看所实现的功能有没有问题