1.JDK(JRE的超集,提供了更多工具如编译器和诊断工具等)>JRE(Java运行环境,包含JVM和Java类库以及一些模块)>JVM
2.Java基于解释执行不准确,Java通过Javac编译为字节码,在运行时通过JVM内嵌的解释器将字节码转化为机器码。但有一种动态编译器能在运行时将热点代码编译为机器码,这就属于编译执行而不是解释执行。JDK8实际是解释和编译混合的一种模式。
3.Exception和Error都是继承了Throwable类,在Java中只有Throwable类的实例才能被抛出(throw)或捕获(catch)
4.try-catch尽量不要捕获类似Exception这种通用的异常,而是捕获特定异常。
5.try-catch代码会产生额外的性能开销,尽量不要一个大的try包住整段代码。
6.Java每实例化一个Exception,都会对当时的栈进行快照,这是一个相对比较重的操作。如果发生的非常频繁,这个开销可就不能被忽略了。
7.捕捉到异常最好就地解决,实实在在处理。
8.不要在finally代码块处理返回值
9.如无必要,勿用异常
10.final变量产生了某种程度的不可变(immutable)的效果,所以,可以用于保护只读数据,尤其是在并发编程中,因为明确地不能再赋值fnal变量,有利于减少额外的同步开销,也可以省去一些防御性拷贝的必要。(提高性能,但忽略不计)
11.Java没有const,通过static+final实现
12.通过关键字new出来的就是强引用,当JVM空间不足时宁愿抛出OutOfMemoryError运行时错误(OOM)使程序异常中止也不会被GC回收。只要超过了引用的作用域或者显式地将相应(强)引用赋值为 null,就是可以被垃圾收集的了,具体回收时机还是要看垃圾收集策略。
13.软引用生命周期比强引用短,只用当JVM认为内存不足才会去尝试回收软引用指向的对象。软引用通常用来实现内存敏感的缓存。
14.弱引用生命周期比软引用生命周期短,在垃圾回收器扫描内存区域时一旦发现弱引用不管内存空间是否足够都会回收。弱应用同样可用于内存敏感的缓存。
15.幻想引用(虚引用)无法通过引用访问对象去获取对象,幻象引用仅仅是提供了一种确保对象被 fnalize 以后,做某些事情的机制。如果
一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如
果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。
ReferenceQueue queue = new ReferenceQueue ();
PhantomReference pr = new PhantomReference (object, queue);
程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之
前采取一些程序行动。
16.String的拼接剪裁都会产生新的String对象。
17.StringBuffer线程安全(通过synchronized实现),StringBuilder非线程安全。都是继承了AbstractStringBuilder。
18.StringBuffer和StringBuilder内部数组默认是每次+16,如果一开始明确字符串的长度可以先进行初始化,避免扩容的开销,扩容是抛弃原有的数组,而创建新的数组。还要进行arraycopy。
19.OOM(OutOfMemoryError)内存溢出,内存泄漏
20.CGLIB能代理普通的类和接口,原生JDK只能代理接口。
21.Integer和int的区别:
1.Integer的默认初始值是null,而int的默认初试值是0。
2.Integer变量必须先实例,int变量则可以直接使用
22.基本数据类型和引用数据类型做运算时会拆箱.
23.基本数据类型赋值给引用数据类型的时候会装箱.
24.Integer的缓存区是-128到127,意思是当构建出来的值是这个范围的时候是直接从缓存区获取而不是重新生成(int不是这样)。
25.自动拆装箱可以简化代码的书写,简写转化的过程。
26.Vector是线程安全的动态数组,如果数组满了会创建新的数组(原来一倍)并拷贝原数组数据。
27.ArrayList非线程安全,当数组满了扩容50%
28.LinkedList双向链表,不需要调整容量,非线性安全.
29.TreeSet支持顺序访问,但是添加删除包含等操作相对低效(log(n))
30.HashTable线程安全,不支持null键和值
31.HashMap支持null键和值
32.TreeMap是基于红黑树的一种提供顺序访问的Map
33.HashMap的内部结构是数组+链表(Entry<V,V>),如果链表大小超过阈值则会变成树形结构。
34.HashMap的负载因子为0.75,如果一开始能估算出HashMap的大小进行初始化是最好的,以免进行扩容损耗性能(重新计算哈希值存储)
35.ReentrantLock性能比Synchronized高
36.ReentrantLock必须手动获取和释放锁
37.Condition需要与 Lock 联合使用,它的作用就是代替Object的那些监视器方法,Condition 中的await()、signal()和signalAll()方法分别对应着Object的wait()、notify()和notifyAll()方法。
38.自旋锁是竞争锁失败的线程的时候不会再操作系统层面挂起等待,而是做几个空循环(希望能在不久获取到锁),然后就进入临界区。如果还没有获得锁则线程会在操作系统层面挂起。在空循环的好处是自旋的消耗会小于线程阻塞挂起的消耗,适用于锁竞争不激烈的情况。如果锁竞争激烈导致线程在获取锁之前一直自旋占用CPU做无用功,造成CPU浪费.
39.自旋锁只在多核CPU有效,单核CPU毫无效果(自旋锁获取锁不成功会一直重试,导致一直占用CPU ,而其他线程不能运行而导致当前线程无法释放锁)。
40.通常弱引用都会和引用队列配合清理机制使用,但是ThreadLocal是个例外
41.可以建立一个线程扫描死锁情况,Java的API有ThreadMXBean,通过findDeadlockedThreads()方法定位。
Java核心基数36讲学习笔记
最后编辑于 :
©著作权归作者所有,转载或内容合作请联系作者
- 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
- 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
- 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
推荐阅读更多精彩内容
- 1. 谈谈你对Java平台的理解。 Java是一种面向对象的语言,最显著的特性有两个方面: 一个就是一次编译,到处...
- 前言 不知道大家有没有这样一种感觉,程序员的数量井喷了。可能是因为互联网火了,也可能是各家培训机构为我们拉来了大量...
- 本文出自 Eddy Wiki ,转载请注明出处:http://eddy.wiki/interview-java.h...