参考文献:Java内部类和匿名内部类的用法_pan_jinquan的博客-CSDN博客_内部类和匿名内部类
(一)、内部类:内部类包括成员内部类、静态内部类、局部内部类、匿名内部类四种。
- 非静态内部类访问外部属性(静态的or实例的)一般可以省略前缀(除非内部有重名的为了进行区分,那就静态的加上外部类名.前缀,实例的加上外部类名.this.前缀)
- 静态内部类只能访问外部的静态属性(变量and方法)(不能访问实例属性),一般可以省略前缀(除非内部有重名的为了进行区分,那就静态的加上外部类名.前缀)
- 除了局部内部类和匿名内部类这种定义在方法内的内部类,其他的内部类都不能访问方法内的属性。
- 局部内部类和匿名内部类都不能给内部的变量使用访问权限修饰符,因为这两个内部类都属于局部变量。
- 因为在同一个类内,所以访问本类的私有属性。
- 内部类都不能定义static变量/方法。但其实静态内部类其实可以。但一般不会这么用。
- 外部其他类不能访问局部内部类和匿名内部类(因为他们算是一个局部变量,其他类只配调用方法)
访问静态内部类和成员内部类的方式(创建对象):
public class a {
public static void kfc(){
// 静态方法创建成员内部类对象要通过外部类的引用来创建内部类的对象。
a a1 = new a();
a.b b1 = a1.new b();
// 如果在同一个类内 或者 import 包.a.b; 则a.b b1 = ... 可以简写为 b b1 =...
// 注意,静态方法创建成员内部类时 可简化为:
a.b b2 = new a().new b();
// 静态方法创建静态内部类对象
c c1 = new c();
}
public void ak(){
// 实例方法创建成员内部类对象
a.b b1 =new b();
// 如果在同一个类内 或者 import 包.a.b; 则a.b b1 =... 可以简写为 b b1 =...
// 实例方法创建静态内部类对象
c c1 = new c();
// 在实例方法内,可直接创建内部类对象,也可以采用在静态方法中的创建对象方法
}
class b{
}
static class c{
}
}
(二)匿名内部类
匿名内部类也就是没有名字的内部类正因为没有名字,所以匿名内部类只能使用一次,它通常用来简化代码编写。
但使用匿名内部类还有个前提条件:必须继承一个父类或实现一个接口,但最多只能继承一个父类,或实现一个接口。
关于匿名内部类还有如下两条规则:
1)匿名内部类不能是抽象类,因为系统在创建匿名内部类的时候,会立即创建内部类的对象。因此不允许将匿名内部类定义成抽象类。
2)匿名内部类不等定义构造器(构造方法),因为匿名内部类没有类名,所以无法定义构造器,但匿名内部类可以定义实例初始化块。
怎样判断一个匿名类的存在啊?看不见名字,感觉只是父类new出一个对象而已,没有匿名类的名字。
先看段伪代码:
abstract class Father(){
....
}
public class Test{
Father f1 = new Father(){ .... } //这里就是有个匿名内部类
}
一般来说,new 一个对象时小括号后应该是分号,也就是new出对象该语句就结束了。但是出现匿名内部类就不一样,小括号后跟的是大括号,大括号中是该new 出对象的具体的实现方法。因为我们知道,一个抽象类是不能直接new 的,必须先有实现类了我们才能new出它的实现类。上面的伪代码就是表示new 的是Father的实现类,这个实现类是个匿名内部类。
其实拆分上面的匿名内部类可为:
class SonOne extends Father{
... //这里的代码和上面匿名内部类,大括号中的代码是一样的
}
public class Test{
Father f1 = new SonOne() ;
}
先看一个例子,体会一下匿名内部类的用法:
运行结果:eat something
可以看到,我们直接将抽象类Person中的方法在大括号中实现了,这样便可以省略一个类的书写。并且,匿名内部类还能用于接口上:
public class JavaTest2 {
public static void main(String[] args) {
Person per = new Person() {
public void say() {// 匿名内部类自定义的方法say
System.out.println("say方法调用");
}
@Override
public void speak() {// 实现接口的的方法speak
System.out.println("speak方法调用");
}
};
per.speak();// 可调用
per.say();// 出错,不能调用
}
}
interface Person {
public void speak();
}
这里per.speak()是可以正常调用的,但per.say()不能调用,为什么呢?注意Person per = new Person()创建的是Person的对象,而非匿名内部类的对象。其实匿名内部类连名字都没有,你咋实例对象去调用它的方法呢?但继承父类的方法和实现的方法是可以正常调用的,本例子中,匿名内部类实现了接口Person的speak方法,因此可以借助Person的对象去调用。
若你确实想调用匿名内部类的自定义的方法say(),当然也有方法:
(1)类似于speak方法的使用,先在Person接口中声明say()方法,再在匿名内部类中覆写此方法。
(2)其实匿名内部类中隐含一个匿名对象,通过该方法可以直接调用say()和speak()方法;代码修改如下:
public class JavaTest2 {
public static void main(String[] args) {
new Person() {
public void say() {// 匿名内部类自定义的方法say
System.out.println("say方法调用");
}
@Override
public void speak() {// 实现接口的的方法speak
System.out.println("speak方法调用");
}
}.say();// 直接调用匿名内部类的方法
}
}
interface Person {
public void speak();
}



