类修饰符
1、内部类可以是静态static的,也可用 public,default,protected 和 private 修饰。
2、与第一条相比,普通类只能由public或默认访问权限修饰符(类名前没有访问权限修饰符)。
内部类的作用
使用内部类可以给我们带来以下优点:
内部类可以很好的实现隐藏(一般的非内部类,是不允许有 private 与 protected 权限的,但内部类可以);
内部类拥有外围类的所有元素的访问权限;
可以实现多重继承;
可以避免修改接口而实现同一个类中两种同名方法的调用: 考虑这样一种情形,一个类要继承一个类,还要实现一个接口,可是它所继承的类和接口里面有两个相同的方法(方法签名一致),那么我们该怎么区分它们呢?这就需要使用内部类了。
内部类的种类
1、成员内部类:成员内部类是外围类的一个成员,是依附于外围类的,所以,只有先创建了外围类对象才能够创建内部类对象。也正是由于这个原因,成员内部类也不能含有 static 的变量和方法;
- 包含 static final 域,但该域的初始化必须是一个常量表达式;
- 内部类可以继承含有static成员的类。
2、静态内部类:静态内部类,就是修饰为 static 的内部类,该内部类对象 不依赖于外部类对象,就是说我们可以直接创建内部类对象,但其只可以直接访问外部类的所有静态成员和静态方法;
3、局部内部类:局部内部类和成员内部类一样被编译,只是它的 作用域发生了改变,它只能在该方法和属性中被使用,出了该方法和属性就会失效;
如果定义一个局部内部类,并且希望它的方法可以直接使用外部定义的数据,那么我们必须将这些数据设为是 final 的;特别地,如果只是局部内部类的构造器需要使用外部参数,那么这些外部参数就没必要设置为 final。
4、匿名内部类:定义匿名内部类的前提是,内部类必须要继承一个类或者实现接口,格式为 new 父类或者接口(){定义子类的内容(如函数等)}。也就是说,匿名内部类最终提供给我们的是一个匿名子类的对象。
- 匿名内部类是没有访问修饰符的;
- 匿名内部类是没有构造方法的 (因为匿名内部类连名字都没有);
- 定义匿名内部类的前提是,内部类必须是继承一个类或者实现接口,格式为 new 父类或者接口(){子类的内容(如函数等)}。也就是说,匿名内部类最终提供给我们的是一个匿名子类的对象;
- 若匿名内部类 (匿名内部类没有构造方法) 需要直接使用其所在的外部类方法的参数时,该形参必须为 final 的;如果匿名内部类没有直接使用其所在的外部类方法的参数时,那么该参数就不必为final 的:
// 情形 1:匿名内部类直接使用其所在的外部类方法的参数 name
public class Outer {
public static void main(String[] args) {
Outer outer = new Outer();
Inner inner = outer.getInner("Inner", "gz");
System.out.println(inner.getName());
}
public Inner getInner(final String name, String city) { // 形参 name 被设为 final
return new Inner() {
private String nameStr = name; // OK
private String cityStr = city; // Error: 形参 city 未被设为 final
public String getName() {
return nameStr;
}
};
}
}
// 情形 2:匿名内部类没有直接使用其所在的外部类方法的参数
public class Outer {
public static void main(String[] args) {
Outer outer = new Outer();
Inner inner = outer.getInner("Inner", "gz");
System.out.println(inner.getName());
}
//注意这里的形参city,由于它没有被匿名内部类直接使用,而是被抽象类Inner的构造函数所使用,所以不必定义为final
public Inner getInner(String name, String city) {
return new Inner(name, city) { // OK,形参 name 和 city 没有被匿名内部类直接使用
private String nameStr = name;
public String getName() {
return nameStr;
}
};
}
}
abstract class Inner {
Inner(String name, String city) {
System.out.println(city);
}
abstract String getName();
}
特别地,匿名内部类通过实例初始化 (实例语句块主要用于匿名内部类中),可以达到类似构造器的效果。