以前看<Java编程思想>的时候,看到过嵌套类跟内部类的区别,不过后来就把它们的概念给忘了吧。昨天在看<数据结构与算法分析(Java语言版)>的时候,又遇到了这个概念,当时就很大的疑惑:嵌套类跟内部类有什么区别?只有是否有关键字static的区别吗?
所以今天找了个时间查了一下两者的详细区别,总结在这篇博客中,既方便自己的复习和学习,也启示他人吧。
1,概念:
定义在一个类内部的类,叫作“嵌套类”。嵌套类分为两种:static的和非static的。后者又有一个专门的名字,叫作“内部类”。所以从概念可以看出,嵌套类跟内部类是所属关系,后者包含于前者。示例代码如下:
1 class OuterClass { 2 ... 3 static class StaticNestedClass { 4 ... 5 } 6 class InnerClass { 7 ... 8 } 9 }
同时,嵌套类是其所在类的成员。内部类可以访问所在类的所有成员,即使该成员是private的。而static嵌套类则不得访问所在类的成员。同时,嵌套类,static和非static的,都可以被声明为private、public、protected和default的。
2,为什么要使用嵌套类?
好处应该都比较文本化吧,以后在使用的过程中去理解和体会吧:对只在一个地方使用的类进行逻辑上的分组;增加了封装性;易于阅读和维护。
3,static嵌套类:
因为static嵌套类不能直接访问所在类的非static成员变量和方法,所以static嵌套类必须通过绑定所在类的实例来进行访问。而对于所在类的静态成员和方法包括private、protected和public的,可以访问。因为它也有static修饰。
static嵌套类通过写出封装的类名来进行实例化和访问其内部成员:
1 OuterClass.StaticNestedClass nestedObject = 2 new OuterClass.StaticNestedClass();
4,内部类:
因为内部类是所在类的成员,所以它可以访问所在类的任意变量和方法,但是它本身却不能定义任何static的变量或方法。
同时,内部类的实例化方式也与static嵌套类有所不同:
1 OuterClass outerObject=new OuterClass(); 2 OuterClass.InnerClass innerObject = outerObject.new InnerClass();
5,内部类的分类:
以前曾经接触过内部类的分类,这里一并总结一下:
以前的所谓的一些面试宝典里面差不多都是将内部类分为四个种类:
静态内部类(既static嵌套类)、成员内部类(既上述内部类)、局部内部类和匿名内部类。前两者都已经介绍过了,下面专门看一下后面两者。
5.1,局部内部类:
定义在方法内部的类叫作“局部内部类”。它的作用域仅限于方法作用域内,只能在方法的作用域内定义和实例化,是用处最小的类类型。和局部变量一样,它不能被修饰为private, public, protected和static的,并且只能访问方法内部定义的final变量。
1 class LocalInner 2 { 3 int a = 1; 4 5 public void doSomething() 6 { 7 int b = 2; 8 final int c = 3; 9 // 定义一个局部内部类 10 class Inner3 11 { 12 public void test() 13 { 14 System.out.println("Hello World"); 15 System.out.println(a); 16 17 // 不可以访问非final的局部变量 18 // error: Cannot refer to a non-final variable b inside an inner 19 // class defined in a different method 20 // System.out.println(b); 21 22 // 可以访问final变量 23 System.out.println(c); 24 } 25 } 26 27 // 创建局部内部类的实例并调用方法 28 new Inner3().test(); 29 } 30 } 31 32 public class LocalInnerClassTest 33 { 34 public static void main(String[] args) 35 { 36 // 创建外部类对象 37 LocalInner inner = new LocalInner(); 38 // 调用外部类的方法 39 inner.doSomething(); 40 } 41 42 }
5.2,匿名内部类:
顾名思义,匿名内部类就是没有名字的局部类。它不使用关键字class, extends, implements以及构造函数。
它通常作为方法的一个参数传入,比如在android开发中对一个Button添加一个OnClickListener监听器。
匿名内部类隐匿的继承了一个父类或者实现了一个接口。比如:
1 mUiHandler.post(new Runnable{ 2 @override 3 public void run(){ 4 // 5 } 6 7 }); 8 9 AsyncClient.get(url, new JsonHttpResponseHandler() { 10 @Override 11 public void onSuccess(int statusCode, Header[] headers, 12 JSONObject response) { 13 // TODO Auto-generated method stub 14 super.onSuccess(statusCode, headers, response);}} );
内部类通过将相关的类组织在一直,从而降低了命名空间的复杂性。
6,内部类的序列化问题。
对任何种类内部类(包括局部内部类和匿名内部类)的序列化都是不被鼓励的。因为java编译器在对内部类进行编译的时候,将进行“合成构造”。合成构造使得java编译器实现了java的新特性,但是却没有对JVM做出相应的改变。然而,不同的java编译器对合成构造是有差别的,因而,如果对内部类进行了序列化,将使得不同的JRE实现中存在兼容性问题。
本文是在参考了大量他人的劳动成果之上的而写成的,主要的参考文献有:
http://docs.oracle.com/javase/tutorial/java/javaOO/nested.html;
http://www.cnblogs.com/mengdd/archive/2013/02/08/2909307.html。