内部类并不经常使用,并且使用起来有必定的定式,好比在下面的InnterDemoByTrhead.java里,咱们经过内部类的形式建立线程。 java
1 public class InnerDemoByThread { 2 public static void main(String[] args) { 3 // 实现runnable接口,建立10个线程并启动 4 for(int threadCnt = 0;threadCnt<10;threadCnt++) 5 new Thread(new Runnable() { 6 public void run() { 7 for (int i = 0; i < 5; i++) { 8 //在每一个线程里,输出0到4 System.out.println(Thread.currentThread().getName()+":"+ i); 9 } 10 } 11 }).start();//这里的括号是和第5行对应,注意须要带分号 12 } 13 }
在上述的第4行里,咱们经过for循环建立了10个线程,在第5行里,咱们经过new Runnable定义了线程内部的动做,具体而言,在第6到第10行的代码里,定义了打印0到4的动做。这里第5行经过new Thread定义的类,是在第1行定义的InnerDemoByThread类的内部,因此叫内部类,这也是内部类典型的用法。面试
虽然内部类出现的机会很少,但其中有个很是重要的知识点:当方法的参数须要被内部类使用时,那么这个参数必须是final,不然会报语法错误。咱们在讲线程的时候,经过内部类比较了线程安全和不安全集合的表现。这里咱们经过改写这个案例,着重看下“内部类“和“final“的要点,请你们看下以下的InnerFinalDemo.java代码。 安全
1 import java.util.ArrayList; 2 import java.util.List; 3 public class InnerFinalDemo { 4 public static int addByThreads(final List list) { 5 // 建立一个线程组 6 ThreadGroup group = new ThreadGroup("Group"); 7 // 经过内部类的方法来建立多线程 8 Runnable listAddTool = new Runnable() { 9 public void run() {// 在其中定义线程的主体代码 10 list.add("0"); // 在集合里添加元素 11 } 12 }; 13 // 启动10个线程,同时向集合里添加元素 14 for (int i = 0; i < 10; i++) { 15 new Thread(group, listAddTool).start(); 16 } 17 while (group.activeCount() > 0) { 18 try { Thread.sleep(10); } 19 catch (InterruptedException e) 20 { e.printStackTrace(); } 21 } 22 return list.size(); // 返回插入后的集合长度 23 } 24 public static void main(String[] args) { 25 List list = new ArrayList(); 26 //很大可能返回10 27 System.out.println(addByThreads(list)); 28 } 29 }
这段代码的逻辑是,在main函数的第25行里,咱们建立了一个线程不安全的ArrayList类型的对象,并在第27行调用了addByThreads方法返回list的长度。在addByThreads方法里,咱们在第14行里,经过for循环启动了10个线程,在这10个线程的主体逻辑(第9行的run方法)里,咱们在第10行经过list.add方法给集合对象添加元素。多线程
从功能上讲,第27行的打印语句能输出10,由于虽然ArrayList是线程不安全对象,但仅仅是10个线程同时操做,不足以发生“线程抢占”的状况。函数
但本代码的重点是内部类和final,在代码第3行定义的addByThreads方法里,咱们注意到参数list前必定得加final,不然会报语法错误。咱们能够经过以下的思惟步骤来理解这个要点。线程
第一,第3行的这个带final的list对象从属于外部的InnerFinalDemo类,而且,在第8到12行的内部类里,也会用到这个对象,也就是说,在外部类和内部类里,都会用到这个对象。指针
第二,外部类和内部类是平行的,内部类并不从属于外部类,这句话隐藏的含义是,外部类有可能在内部类以前被回收。对象
那么若是咱们不加final,一旦外部类在内部类以前被回收,那么外部类里所包含的list对象也会被回收,但这时,内部类还没有使用这个list。在这种状况下,一旦内部类使用了list,就会报空指针错(由于这个对象已经随着外部类被回收了)。blog
为了不这种错误,在指定语法时就加上了“当方法的参数须要被内部类使用时,那么这个参数必须是final”这个规定。一旦在此类参数前加final,那么这个参数就是常量了,存储的位置就不是“堆区”了,而是“常量池”,这样即便外部类被先回收,那么因为这类参数(好比list)不存在于外部类所从属的堆空间(而是常量池),因此会继续存在,这样内部类就能继续使用。接口
一些资深的面试官不会面试内部类的细节语法(由于不经常使用,并且使用起来有定式),而会考察上述的“参数和final”的知识点,因此你们在被问及”对内部类的掌握程度“这类问题时,能够按以下的思路来叙述。
第一,无需叙述内部类中各类语法,事实上,内部类涉及到“如何定义”以及“内部类中对象的可见性”等问题,语法相对而言比较复杂,提及来不容易,并且即便说清楚了,也没法很好体现你们的能力。
第二,能够直接说,“当方法的参数须要被内部类使用时,那么这个参数必须是final”,同时解释下缘由。当面试官听到这之后,通常就再也不问内部类问题了,由于他会认为,候选人连这么“资深”的知识也知道,那么就不必再细问内部类的问题了。
第三,因为已经引出“垃圾回收”的话题,因此你们能够找机会进一步按本章给出的提示,展现在这方面的能力,这样就有很大可能获得“Java Core方面比较资深”的评价。
上述叙述是针对jdk1.7以及以前版本的,若是是针对jdk1.8版本,不须要显式地加final,但依然会被当常量管理,具体来说,该对象的引用没法指向新的内存空间。