内部类的声明与访问
内部类可以是static的也可以用其它四种访问修饰符(而外部类只能使用public和 default修饰)。内部类所能实现的功能外部类也同样可以全部实现,只是使用内部类可以使其更加简洁高效。
内部类是一个编译时概念,一旦编译成功就会变成两个完全不同的类,对于一个名为outer的外部类和一个名为inner的内部类。编译结束之后会出现outer.class和outer$inner.class两类。所以外部类的属性/方法名可以和内部类完全一样。同时内部类可以访问外部类全部的属性和方法(包括private修饰的),但如果是具有相同命名的属性和方法则需要一定修饰:外部内名.this.外部成员名;同时如果要创建内部类对象不能直接new一个内部类,必须要通过外部类。比如要创建一个内部类iner对象,需要这么做:
Outer outer = new Outer();
Outer.Inner iner = outer.new Inner();
package cn.chujian6.anonymityclass;
public class InnerTest {
public static void main(String[] args) {
Outer outer = new Outer();
outer.test();
System.out.println(outer.getI());
System.out.println("-------1--------");
Outer.Inner iner = outer.new Inner();
iner.innerMsg();
System.out.println(iner.getI());
System.out.println("-------2--------");
System.out.println(outer.getI());
}
}
class Outer {
private int i = 10;
private int y = 8;
public Outer() {
System.out.println("调用Outer构造方法:outer");
}
public void sayMsg() {
System.out.println("Outer class!");
}
class Inner {
int i = 1000;
Inner() {
System.out.println("调用Inner构造方法:inner");
//可以直接访问外部类
System.out.println(y);
}
void innerMsg() {
System.out.println(">>>>>Inner class!");
sayMsg();
//访问内部类自己的成员i,也可以写成 this.i++
this.i++;
//访问外部类的成员 i和y
Outer.this.i++;
y--;
}
int getI() {
return i;
}
}
public void test() {
Inner in = new Inner();
in.innerMsg();
}
public int getI() {
return i;
}
public void setI(int i) {
this.i = i;
}
}
局部内部类
局部类不能用public或private访问说明符进行声明。它的作用域被限定在声明这个局部类块中。局部类有一个优势,他可以对外部完全隐藏起来。Thinking in java中给出了这个例子:
//定义在方法内
public class Parcel4 {
public Destination destination(String s) {
class PDestination implements Destination {
private String label;
private PDestination(String whereTo) {
label = whereTo;
}
public String readLabel() {
return label;
}
}
return new PDestination(s);
}
public static void main(String[] args) {
Parcel4 p = new Parcel4();
Destination d = p.destination("Tasmania");
}
}
//定义在作用域内
public class Parcel5 {
private void internalTracking(boolean b) {
if (b) {
class TrackingSlip {
private String id;
TrackingSlip(String s) {
id = s;
}
String getSlip() {
return id;
}
}
TrackingSlip ts = new TrackingSlip("slip");
String s = ts.getSlip();
}
}
public void track() {
internalTracking(true);
}
public static void main(String[] args) {
Parcel5 p = new Parcel5();
p.track();
}
}
局部类和普通类相似,只是作用域不同而已,它只能在该方法或者条件的作用域类有效。
静态内部类
有的时候内部类只是为了把一个类隐藏到另外一个类的内部,并不需要内部类引用外部类的对象。因此声名为static,以便取消产生的引用。
在内部类不需要访问外围类对象的时候,应该使用静态内部类。
与常规类不同,静态内部类可以有静态方法和静态域。
匿名内部类
匿名内部类是没有名称的类,所以没有办法引用他们,只能在创建的时候作为new语句的一部分来声明它。形式如下:new (类或接口){类的主体};这种形式的new语句声明的一个新的匿名内部类,它完成了对给定类的扩展或者是实现了一个给定接口,同时还创建了那个类的实例,并把它作为语句的结果返回。
加入这个匿名类是对一个类进行扩展,它可以访问这个类的属性和方法等,和其它标准类相同;如果是实现了一个给定的接口,那就必须要实现接口的方法。注意匿名类声明是在编译时运行,实例化是在运行时进行。
有一点需要注意,由于匿名类没有名字,所以他们没有构造方法,但是如果它继承了一个没有无参构造器的类,创建它的时候就必须带上这些参数,并且在使用过程中调用super关键字调用相应的内容。如果想初始化它的成员变量,有几种方法:
1.如果是一个方法的匿名内部类,可以利用这个方法传进你想要的参数,这些参数必须使用final修饰;
2.将匿名内部类改造成有名字的局部内部类,这样就可以有构造方法了;
3.在这个匿名内部类中使用初始化块。
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并没有被使用,所以不需要final
public Inner getInner(final String name, String city) {
return new Inner(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();
}
留意外部类的方法的形参,当所在的方法的形参需要被内部类里面使用时,该形参必须为final。这里可以看到形参name已经定义为final了,而形参city 没有被使用则不用定义为final。为什么要定义为final呢?在网上找到的解释:
“这是一个编译器设计的问题,如果你了解java的编译原理的话很容易理解。首先,内部类被编译的时候会生成一个单独的内部类的.class文件,这个文件并不与外部类在同一class文件中。当外部类传的参数被内部类调用时,从java程序的角度来看是直接的调用例如:
public void dosome(final String a,final int b){
class Dosome{public void dosome(){System.out.println(a+b)}};
Dosome some=new Dosome();
some.dosome();
}
从代码来看好像是那个内部类直接调用的a参数和b参数,但是实际上不是,在java编译器编译以后实际的操作代码是
class Outer$Dosome{
public Dosome(final String a,final int b){
this.Dosome$a=a;
this.Dosome$b=b;
}
public void dosome(){
System.out.println(this.Dosome$a+this.Dosome$b);
}
}}
从以上代码看来,内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到了自己的内部,自己内部的方法调用的实际是自己的属性而不是外部类方法的参数。这样理解就很容易得出为什么要用final了,因为两者从外表看起来是同一个东西,实际上却不是这样,如果内部类改掉了这些参数的值也不可能影响到原参数,然而这样却失去了参数的一致性,因为从编程人员的角度来看他们是同一个东西,如果编程人员在程序设计的时候在内部类中改掉参数的值,但是外部调用的时候又发现值其实没有被改掉,这就让人非常的难以理解和接受,为了避免这种尴尬的问题存在,所以编译器设计人员把内部类能够使用的参数设定为必须是final来规避这种莫名其妙错误的存在。” (简单理解就是,拷贝引用,为了避免引用值发生改变,例如被外部类的方法修改等,而导致内部类得到的值不一致,于是用final来让该引用不可改变)。
因为匿名内部类,没名字,是用默认的构造函数的,无参数的,那如果需要参数呢?则需要该类有带参数的构造函数:
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) {
return new Inner(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();
}
注意这里的形参city,由于它没有被匿名内部类直接使用,而是被抽象类Inner的构造函数所使用,所以不必定义为final。
而匿名内部类通过实例初始化,可以达到类似构造器的效果:
public class Outer {
public static void main(String[] args) {
Outer outer = new Outer();
Inner inner = outer.getInner("Inner", "gz");
System.out.println(inner.getName());
System.out.println(inner.getProvince());
}
public Inner getInner(final String name, final String city) {
return new Inner() {
private String nameStr = name;
private String province;
// 实例初始化
{
if (city.equals("gz")) {
province = "gd";
}else {
province = "";
}
}
public String getName() {
return nameStr;
}
public String getProvince() {
return province;
}
};
}
}
interface Inner {
String getName();
String getProvince();
}