Exception又分为了运行时异常和编译时异常。
编译时异常(受检异常)表示当前调用的方法体内部抛出了一个异常,所以编译器检测到这段代码在运行时可能会出异常,所以要求我们必须对异常进行相应的处理,可以捕获异常或者抛给上层调用方。
运行时异常(非受检异常)表示在运行时出现的异常,常见的运行时异常包括:空指针异常,数组越界异常,数字转换异常以及算术异常等。
异常Exception应该被捕获,使用try – catch – finally 来处理异常,并且使得程序恢复正常。
捕获异常应该遵循哪些原则呢
不遵守这些原则的话,捕获到异常,却看不出是哪里抛出的!
一个容易混淆的异常与错误知识点
NoClassDefFoundError 和 ClassNotFoundException 有什么区别?
Java是一种先编译,后解释执行的语言。何为JIT编译器呢?
JIT编译器全名叫Just In Time Compile 也就是即时编译器,把经常运行的代码作为"热点代码"编译成与本地平台相关的机器码,并进行各种层次的优化。JIT编译除了具有缓存的功能外,还会对代码做各种优化,包括逃逸分析、锁消除、 锁膨胀、方法内联、空值检查消除、类型检测消除以及公共子表达式消除等。
逃逸分析
逃逸分析的基本行为就是分析对象动态作用域,当一个对象在方法中被定义后,它可能被外部方法所引用,例如作为调用参数传递到其他地方中,称为方法逃逸。
标量和聚合量的基本概念
在JIT阶段,如果经过逃逸分析,发现一个对象不会被外界访问的话,那么经过JIT优化,就会把这个对象拆解成若干个其中包含的若干个成员变量来代替。这个过程就是标量替换。标量替换的好处就是对象可以不在堆内存进行分配,为栈上分配提供了良好的基础。
逃逸分析缺点:技术不是特别成熟,分析的过程也很耗时,如果没有一个对象是不逃逸的,那么就没用了。
值传递案例
public class Test { public static void main(String[] args) { int x=0; change(x); System.out.println(x); } static void change(int i){ i=7; } }
上边的代码会输出0。因为如果参数是基本数据类型,那么是属于值传递的范畴,传递的其实是源对象的一个copy副本,不会影响源对象的值。
引用传递案例
public class Test { public static void main(String[] args) { StringBuffer x = new StringBuffer("Hello"); change(x); System.out.println(x); } static void change(StringBuffer i) { i.append(" world!"); } }
输出为Hello world!
变化一下
public class Test { public static void main(String[] args) { StringBuffer x = new StringBuffer("Hello"); change2(x); System.out.println(x); } static void change2(StringBuffer i) { i = new StringBuffer("hi"); i.append(" world!"); } }
再变化一下
public class Test { public static void main(String[] args) { StringBuffer sb = new StringBuffer("Hello "); System.out.println("Before change, sb = " + sb); changeData(sb); System.out.println("After change, sb = " + sb); } public static void changeData(StringBuffer strBuf) { StringBuffer sb2 = new StringBuffer("Hi,I am "); strBuf = sb2; sb2.append("World!"); } }
同样画内存图不难得到