内部类并不常用,而且使用起来有一定的定式,比如在下面的InnterDemoByTrhead.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,就会报空指针错(因为这个对象已经随着外部类被回收了)。

为了避免这种错误,在指定语法时就加上了“当方法的参数需要被内部类使用时,那么这个参数必须是final”这个规定。一旦在此类参数前加final,那么这个参数就是常量了,存储的位置就不是“堆区”了,而是“常量池”,这样即使外部类被先回收,那么由于这类参数(比如list)不存在于外部类所从属的堆空间(而是常量池),所以会继续存在,这样内部类就能继续使用。

一些资深的面试官不会面试内部类的细节语法(因为不常用,而且使用起来有定式),而会考察上述的“参数和final”的知识点,所以大家在被问及”对内部类的掌握程度“这类问题时,可以按如下的思路来叙述。

第一,无需叙述内部类中各种语法,事实上,内部类涉及到“如何定义”以及“内部类中对象的可见性”等问题,语法相对而言比较复杂,说起来不容易,而且即使说清楚了,也无法很好体现大家的能力。

第二,可以直接说,“当方法的参数需要被内部类使用时,那么这个参数必须是final”,同时解释下原因。当面试官听到这以后,一般就不再问内部类问题了,因为他会认为,候选人连这么“资深”的知识也知道,那么就没必要再细问内部类的问题了。

第三,由于已经引出“垃圾回收”的话题,所以大家可以找机会进一步按本章给出的提示,展示在这方面的能力,这样就有很大可能得到“Java Core方面比较资深”的评价。

上述叙述是针对jdk1.7以及之前版本的,如果是针对jdk1.8版本,不需要显式地加final,但依然会被当常量管理,具体来讲,该对象的引用无法指向新的内存空间。

  

内部类、final与垃圾回收,面试时你一说,面试官就知道的更多相关文章

  1. 推荐收藏系列:一文理解JVM虚拟机(内存、垃圾回收、性能优化)解决面试中遇到问题(图解版)

    欢迎一起学习 <提升能力,涨薪可待篇> <面试知识,工作可待篇 > <实战演练,拒绝996篇 > 欢迎关注我博客 也欢迎关注公 众 号[Ccww笔记],原创技术文章 ...

  2. Java垃圾回收机制你还不明白?一线大厂面试必问的!

    什么是自动垃圾回收? 自动垃圾回收是一种在堆内存中找出哪些对象在被使用,还有哪些对象没被使用,并且将后者删掉的机制. 所谓使用中的对象(已引用对象),指的是程序中有指针指向的对象:而未使用中的对象(未 ...

  3. JVM系列之五:垃圾回收

    . jdk1.7的堆内存 1. 堆(Java堆) 堆是java虚拟机所管理的内存中最大的一块内存区域,也是被各个线程共享的内存区域, 在JVM启动时创建,该内存区域存放了对象实例(包括基本类型的变量及 ...

  4. Java Garbage Collection基础详解------Java 垃圾回收机制技术详解

    最近还是在找工作,在面试某移动互联网公司之前认为自己对Java的GC机制已经相当了解,其他面试官问的时候也不存在问题,直到那天该公司一个做搜索的面试官问了我GC的问题,具体就是:老年代使用的是哪中垃圾 ...

  5. JVM的stack和heap,JVM内存模型,垃圾回收策略,分代收集,增量收集

    (转自:http://my.oschina.net/u/436879/blog/85478) 在JVM中,内存分为两个部分,Stack(栈)和Heap(堆),这里,我们从JVM的内存管理原理的角度来认 ...

  6. 管理Java垃圾回收的五个建议

    [编者按]本文作者是Niv Steingarten,是Takipi 的联合创始人,热衷于编写优雅简洁的代码.作者通过对垃圾收集器的介绍和梳理,在管理垃圾回收方面提出了五个建议,降低收集器开销,帮助大家 ...

  7. Golang垃圾回收机制(二)

    原文:https://blog.csdn.net/qq_15427331/article/details/54613635 Go语言正在构建的垃圾收集器(GC),似乎并不像宣传中那样的,技术上迎来了巨 ...

  8. Java 清理和垃圾回收

    java.lang.ref.cleaner包 finalize()//该方法已过时,有风险,慎用 1.对象不可能被垃圾回收 2.垃圾回收并不等于"析构" 只有当垃圾回收发生时fin ...

  9. java虚拟机垃圾回收机制详解

    首先,看一下java虚拟机运行的时候内存分配图: jvm虚拟机栈:一个是线程独有的,每次启动一个线程,就创建一个jvm虚拟机栈,线程退出的时候就销毁.这里面主要保存线程本地变量名和局部变量值. 本地方 ...

随机推荐

  1. <Standard Template Library>标准模板库专项复习总结(一)

    看了看博客园的申请时间也一年多了...想想自己一年多以来一直处于各种划水状态,现在又要面临ACM的冲击... 还是要抓紧时间赶紧复习一下了- -毕竟校园新生赛还是有奖金的.. 1.栈 先进后出(LIF ...

  2. ASP.NET Core 连接 GitLab 与 MatterMost 打造 devops 工具

    在现代化开发工具链里面就包含了自动化的通讯工具,而日志写代码我是推到 Gitlab 平台上,我今天听了郭锐大佬的分享之后,感觉我现在的团队的自动化做的远远不够.我在他的课程上学到的最重要一句话就是做工 ...

  3. Trie 树的一些题

    Trie 树的一些题 牛客练习赛11 假的字符串 (Trie树+拓扑找环) 链接:https://ac.nowcoder.com/acm/problem/15049 来源:牛客网 给定n个字符串,互不 ...

  4. 使用FluentEmail发送outlook邮件

    一,邮箱账号相关设置 1,创建outLook邮箱. 2,进入邮箱设置->同步电子邮件->允许设备和应用使用pop 3,设置microsoft账号的应用程序密码->进入安全性页面-&g ...

  5. android权限申请执行过程

    弹出的授权对口窗口实际上是一个activity(隐式intent启动上图第二个方块里)(com.android.packageinstaller.permission.ui.Grantpermissi ...

  6. Linux基础:认识Linux

    1.Linux操作系统的特点 优点 ​ (1)可靠性高:linux是基于Unix的概念开发出来的系统,拥有Unix的稳定且效率的特点.运行一年以上而不曾宕机.不必关机是很平常的事情 : ​ (2)彻底 ...

  7. 怎么安装GUI

    python安装easygui的过程中,下载的是0.97.安装的时候提示setuptools模块不存在.然后又去安装setuptools等等, 真麻烦.也没有成功.后来又下载了0.96的.才成功.下面 ...

  8. Java线程的生命周期与状态流转

    上图是一个线程的生命周期状态流转图,很清楚的描绘了一个线程从创建到终止的过程. 这些状态的枚举值都定义在java.lang.Thread.State下 NEW:毫无疑问表示的是刚创建的线程,还没有开始 ...

  9. Linux-Cacti监控{Verson:1.2.8}

    首先需要一个LAMP平台 或LNMP平台 yum -y install httpd mariadb php mariadb-server mariadb-devel zlib freetype lib ...

  10. 《提升能力,涨薪可待》—Java并发之Synchronized

    Synchronized简介 线程安全是并发编程中的至关重要的,造成线程安全问题的主要原因: 临界资源, 存在共享数据 多线程共同操作共享数据 而Java关键字synchronized,为多线程场景下 ...