java内部类使用小结
2016-12-23 16:19
459 查看
概述
内部类是指在一个外部类的内部再定义一个类。类名不需要和文件夹相同。内部类可以是静态static的,也可用public,default,protected和private修饰,而外部顶级类即类名和文件名相同的只能使用public和default
内部类是一个编译时的概念,一旦编译成功,就会成为完全不同的两类。对于一个名为outer的外部类和其内部定义的名为inner的内部类。编译完成后出现outer.class和outer$inner.class两类。所以内部类的成员变量/方法名可以和外部类的相同。
内部类主要可分为成员内部类、局部内部类、嵌套内部类、匿名内部类等
成员内部类
成员内部类,就是作为外部类的成员,可以直接使用外部类的所有成员和方法,即使是private的。同时外部类要访问内部类的所有成员变量/方法,则需要通过内部类的对象来获取;要注意的是,成员内部类不能含有static的变量和方法。因为成员内部类需要先创建了外部类,才能创建它自己的,了解这一点,就可以明白更多事情,在此省略更多的细节了;
在成员内部类要引用外部类对象时,使用outer.this来表示外部类对象;
而需要创建内部类对象,可以使用outer.inner obj = outerobj.new inner()。
public class Outer { public class Inner { public void say(String str) { System.out.println(str); } } //个人推荐使用getxxx()来获取成员内部类,尤其是该内部类的构造函数无参数时 public Inner getInner() { return new Inner(); } public static void main(String[] args) { Outer outer = new Outer(); Inner inner = outer.new Inner(); inner.say("hello world!"); Inner inner2 = outer.getInner(); inner2.say("你好,世界!"); } }
局部内部类
局部内部类,是指内部类定义在方法和作用域内。请看如下两个例子:定义在方法内:
public class Outer { interface Spider { void fly(); } public Spider getSpiderman() { //定义局部内部类 class Spiderman implements Spider { @Override public void fly() { System.out.println("我是超人我会飞!"); } } //使用局部内部类 return new Spiderman(); } public static void main(String[] args) { Spider spider = new Outer().getSpiderman(); spider.fly(); } }
定义在作用于内:
public class Outer { private static final String NOTICE = "我是超人我会飞!"; private void spidermanSayHello(boolean flag) { if (flag) { //定义局部内部类 class Spiderman { public void fly() { System.out.println(NOTICE); } } //使用局部内部类 new Spiderman().fly(); } } public void spidermanSayHello() { spidermanSayHello(true); } public static void main(String[] args) { Outer outer = new Outer(); outer.spidermanSayHello(); } }
局部内部类也像别的类一样进行编译,但只是作用域不同而已,只在该方法或条件的作用域内才能使用,退出这些作用域后无法引用的。
嵌套内部类
嵌套内部类,就是修饰为static的内部类。声明为static的内部类,不需要内部类对象和外部类对象之间的联系,就是说我们可以直接引用outer.inner,即不需要创建外部类,也不需要创建内部类。嵌套类和普通的内部类还有一个区别:普通内部类不能有static数据和static属性,也不能包含嵌套类,但嵌套类可以。而嵌套类不能声明为private,一般声明为public,方便调用。
public class Outer { public static String STR = "I am in outer"; public static void say() { System.out.println("outer: " + STR); } static class Inner { public static String STR = "I am in inner"; public static void say() { System.out.println(STR); Outer.say(); } } public static void main(String[] args) { String ss = Outer.Inner.STR; System.out.println(ss); //I am in inner Outer.Inner.say(); //I am in inner //outer: I am in outer } }
匿名内部类
有时候我为了免去给内部类命名,便倾向于使用匿名内部类,因为它没有名字。匿名内部类不能加访问修饰符。
/** * 被内部类使用到的外部类的方法的形参(注意city没有被内部类使用到),必须为final。 * 为什么要定义为final呢? * 简单理解就是,在拷贝引用时,为了避免引用值发生改变,例如被外部类的方法修改,而导致内部类得到的值不一致,于是用final来让该引用不可改变。 */ public class Outer { // interface Spider { String getName(); } // public Spider getSpiderman(final String name, String city) { System.out.println("city: " + city); return new Spider() { private String nameStr = name; @Override public String getName() { return nameStr; } }; } public static void main(String[] args) { Outer outer = new Outer(); Spider spiderman = outer.getSpiderman("Parker", "New York"); System.out.println(spiderman.getName()); //Parker } }
内部类的继承
内部类的继承,是指内部类被继承,普通类 extents 内部类。而这时候代码上要有点特别处理,具体看以下例子:public class A extends Outer.Inner { //无参构造函数A()是不能通过编译的,一定要加上形参 A(Outer outer) { outer.super(); } public static void main(String[] args) { Outer outer = new Outer(); A a = new A(outer); a.say("hello!"); } } class Outer { class Inner { public void say(String str) { System.out.println(str); } } }