在计算机程序运行的过程中,经常会出现错误,例如:用户输入错误、读写文件错误、网络错误、内存耗尽等等。所以在计算机程序运行的过程中,错误时不可避免的。
如果我们的方法调用出错,那么调用方该如何知道这个错误呢?
可以约定返回错误码,但是处理它却会很困难,我们需要编写大量的if-else以及switch语句来处理不同的错误。而且由于Java只允许返回一个字符,如果我们既要返回错误码又要返回正常结果,就只能使用JavaBean来进行包装,所以Java引用了一种新的机制用来表示程序中出现的错误,也就是异常。
(1)Java使用异常来表示错误
(2)Java的异常是class,本身带有类型信息,并且从Throwable继承
(3)异常可以在任何地方抛出
(4)异常只需要在上层捕获,与方法调用分离
在下面这段代码中,processFile可能会抛出任何异常,通过catch语句捕获异常
try {
String s = processFile("C:\\test.txt");
//OK
}catch (FileNotFoundException e) {
//file not found:
}catch (SecurityException e) {
//no read permission:
}catch (IOException e) {
//io error:
}catch (Exception e) {
//other error:
}
由于Java的异常也是一种class,因此它也拥有自己的继承体系:
(1)必须捕获的异常:
Java语言规定:我们必须捕获的异常包括Exception以及它的子类,但不包括RuntimeException及其子类(Checked Exception)
(2)不需要捕获的异常:
①Error及其子类
②RuntimeException及其子类
那么Java为什么要这样规定异常的继承体系呢?
(1)Error是指发生了严重的错误,Java程序对此无能为力。例如:
OutOfMemoryError(内存耗尽),NoClassDefFoundError(虚拟机没有找到class文件),StackOverflowError…
(2)Exception是指发生了运行时的逻辑错误,Java程序应当去捕获异常并进行处理。
①某些错误是预期可能会发生的错误,例如:IOException(读写错误),NumberFormatException(用户输入格式错误)…
②某些程序是代码的编写逻辑存在BUG,例如:NullPointException,IndexOutOfBoundsException…
在Java中我们使用try{…}catch(){…}语句捕获异常,将可能发生异常的语句放在try{…}代码块中执行,之后用catch语句捕获对应的Exception及其子类
public static void main(String[] args) {
try {//将可能发生异常的语句放入try代码块中执行
process1();
process2();
process3();
}catch(IOException e) {//使用catch语句捕获对应的异常以及它的子类
System.out.println(e);
}
}
如果某个方法调用抛出Checked Exception,必须捕获这个异常并进行处理,例如:
static byte[] toGBK(String s) {
try {
return s.getBytes("GBK");
}
catch (UnsupportedEncodingException e) {
System.out.println(e);
}
}
如果我们在这个方法中不去捕获这个异常,那么我们就必须通过throws来声明这个异常,通过throws声明的异常在这个异常中没有被捕获,但是它依然需要在上层进行捕获
static byte[] toGBK(String s) {
throws UnsupportedEncodingException {//申明异常暂时不进行捕获
return s.getBytes("GBK");
}
public static void main(String[] args) {
try {//在上层代码进行异常捕获
return s.getBytes("GBK");
}
catch (UnsupportedEncodingException e) {
System.out.println(e);
}
}
所以,通过throws声明异常,只是推迟了异常的处理,最终还是要在某一层捕获这个异常
由于main方法是Java程序调用的第一个代码,因此main()是最后捕获Exception的机会。如果一个Exception被抛出,而在main方法中并没有被捕获到,JVM将会发生报错,并退出程序。
RuntimeException无需强制捕获,非RuntimeException(CheckException)需强制捕获,或者用throws声明
在前面我们了解到,我们可以使用try{…}catch(){…}语句捕获异常,将可能发生异常的语句放在try{…}代码块中执行,之后用catch语句捕获对应的Exception及其子类。
我们还可以使用多个catch子句进行捕获异常
(1)每个catch捕获对应的Exception及其子类
(2)从上到下分配,分配到某个catch后不再继续匹配
例如:
public static void main(String[] args) {
try {//将可能发生异常的语句放入try代码块中执行
process1();
process2();
process3();
}catch(IOException e) {
System.out.println(e);
}catch(NumberForamtException e) {
System.out.println(e);
}
}
(3)catch的分配顺序非常重要,必须将子类写在前面
public static void main(String[] args) {
try {//将可能发生异常的语句放入try代码块中执行
process1();
process2();
process3();
}catch(UnsupportedEncodingException e) {
System.out.println(e);
}catch(IOException e) {
System.out.println(e);
}
}
如何需要编写无论错误是否发生都必须执行的语句,我们需要用到finally语句,finally语句可以保证无论错误是否发生都执行程序。
(1)finally不是必须的
(2)finally总在最后执行
public static void main(String[] args) {
try {//将可能发生异常的语句放入try代码块中执行
process1();
process2();
process3();
}catch(UnsupportedEncodingException e) {
System.out.println(e);
}catch(IOException e) {
System.out.println(e);
}finally {
System.out.println("END");
}
}
如果某些异常的处理逻辑相同,但是并不存在继承关系,就必须编写更多的catch子句,在这种情况下,我们可以使用或操作符“|”将多种异常写在一起,从而简化编写过程,例如:
public static void main(String[] args) {
try {
process1();
process2();
process3();
}catch(IOException | NumberForamtException e) {//合并相同逻辑的异常
System.out.println("Bad input");
}catch(Exception e) {
System.out.println("Unknown error");
}
}
当某个方法抛出异常时,如果当前方法没有捕获,异常就被抛到上层调用方法,直到某个try…catch被捕获。
printStackTrace()可以打印出方法的调用栈,它对于调试错误非常有用。
在Java中,我们需要捕获异常是因为某些方法可以抛出异常。如果我们想要自行抛出异常该如何编写代码呢?
由于异常本身也是一个class,所以,当我们要抛出异常时:
(1)创建某个Exception的实例
(2)用throw语句抛出
例如:如果我们想要抛出IllegalArgumentException,只要将throw语句和创建的实例一起编写就可以了
static void process1(String s) {
throw new IllegalArgumentException();
}
(1)如果一个方法捕获了某个异常之后,又在catch语句中抛出新的异常,就相当于把原来抛出的异常类型转换了
public static void main(String[] args) {
process1("");
}
static void process1(String s) {
try {
process2();
}catch(NullPointException e) {
threw new IllegalArgumentException();
}
}
static void process2(String s) {
threw new NullPointException();
}
但是在这时,新的异常丢失了原始的异常信息,我们只能追踪到IllegalArgumentException,而无法追踪到原来的NullPointException。
因此,我们需要把原有的异常NullPointException传入IllegalArgumentException的实例e,从而让新的Exception可以持有原始类型信息。
public static void main(String[] args) {
process1("");
}
static void process1(String s) {
try {
process2();
}catch(NullPointException e) {
threw new IllegalArgumentException(e);//将原有异常传入实例
}
}
static void process2(String s) {
threw new NullPointException();
}
(2)需要特别注意的是,在抛出异常之前,finally语句会保证执行。例如:
public static void main(String[] args) {
try{
process1("");
}catch(Exception e){//catch Exception
System.out.println("catched");//打印catched
throw new RuntimeException(e);
//在抛出新的异常之前会执行finally语句
}finally{//
System.out.println("finally");//打印出finally
}
}
static void process1(String s) {
threw new IllegalArgumentException();
}
(3)如果finally语句抛出了异常,那么catch语句将不再执行。例如:
public static void main(String[] args) {
try{
process1("");
}catch(Exception e){//catch Exception
System.out.println("catched");
throw new RuntimeException(e);
//由于在finally语句中已经抛出了异常,catch语句中的异常将不会被运行
}finally{//
System.out.println("finally");
throw new NullPointException();//在finally中抛出新的异常
}
}
static void process1(String s) {
threw new IllegalArgumentException();
(4)没有被抛出的异常称为被屏蔽的异常(suppressed exception)
由于Java的异常处理机制使得它只能传播一个异常,后抛出的异常会将前面抛出的但是没有捕获的异常覆盖掉,所以前面抛出的异常就丢失了。
(1)用origin变量保存原始变量
(2)如果存在原始异常用addSuppressed()添加新异常
(3)如果存在原始异常或新异常,最后在finally中抛出
Exception origin = null;
try {
process1("");
}catch (Exception e) {
origin = e;
throw new RuntimeException(e);
}finally {
try {
throw new NullPointException();
}catch(Exception e) {
if(origin != null){
origin.addSuppressed(e);
}else{
origin = e;
}
}
if(origin != null){
throw origin;
}
}
(1)用getSuppressed()获取所有Suppressed Exception
(2)处理Suppressed Exception要求JDK版本>=1.7
try {
somethingWrong("");
}catch(Exception e){
e.printSrackTrace();
for(Throwalbe t : e.getSuppressed()){
t.printSrackTrace();
}
}
(1)JDK定义的常用异常:
——RuntimeException
NullPointerException
IndexOutOfBoundsException
SecurityException
IllegalArgumentException
NumberFormatException
——IOException
UnsupportedCharsetException,FileNotFoundException,SocketException
——ParseException,GeneralSecurityException,SQLException,TimeoutException
(2)当进行抛出异常时,尽量使用JDK已经定义的异常
(3)可以定义新的异常类型:
定义一个自定义的异常类型需要从合适的异常类型中派生,推荐通过RuntimeException派生,这样就不必强制去捕获异常,也不需要在方法中去申明需要抛出的异常
public class BadFileFormatException extends IOException {
}
public class UserNotFoundException extends RuntimeException {
}
(4)可以定义新的异常关系树:
①从合适的Exception中派生BaseException
②其他Exception从BaseException派生
public class BaseException extends RuntimeException {
}
public class UserNotFoundException extends BaseException {
}
public class LoginFailedException extends BaseException {
}
(5)自定义异常应当提供多种构造方法:
提供多个构造方法的目的,是为了在catch语句中,如果我们需要抛出新的BaseException,需要把原有的异常信息传入到BaseException中。
可以使用IDE根据父类快速创建构造方法。
断言是一种调试方式
(1)使用assert关键字
(2)断言条件预期为true
(3)如果运行中结果为false,那么断言失败,将会抛出AssertionError
static double abs(double d) {
return d >= 0 ? d : -d;
}
public static void main(String[] args) {
double x = abs(-123.45);
assert >= 0;
System.out.println(x);
}
(4)可以添加可选的断言消息,在断言失败时,随AssertionError一同抛出
static double abs(double d) {
return d >= 0 ? d : -d;
}
public static void main(String[] args) {
double x = abs(-123.45);
assert >= 0 : "x must >= 0";
System.out.println(x);
}
(1)断言失败时会抛出AssertionError导致程序直接退出
(2)对于可恢复的错误不能使用断言(此时应该抛出异常)
(3)只能在开发和测试阶段启用断言
(4)断言很少使用,更好的方法是使用单元测试
例如:我们不能用assert来判断变量arr是否为空:
void sort(int[] arr) {
assert arr != null;
}
我们应当抛出异常并在上层捕获异常:
void sort(int[] arr) {
if(x == null) {
throw new IllegalArgumentException("arr cannot be null");
}
}
JVM默认情况是关闭断言指导的,会忽略所有的assert语句。
(1)给Java虚拟机传递-ea参数启用断言
(2)可以指定特定的类启用断言:-ea:xxxx.xxxx.xxxx.main(完整类名)
(3)可以指定特定的包启用断言:-ea:xxxx.xxxx(完整包名)
(1)选择Object中的Main.java
(2)选择Run As
(3)选择Run Configurations
(4)选择Arguments
(5)选择VM Arguments
(6)添加启动参数和完整类名-ea:xxxx.xxxx.xxxx.main
(7)保存并运行
(1)取代System.out.println()语句
(2)可以设置输出样式
(3)可以设置输出级别,禁止某些级别的输出
(4)可以被重定向到文件 可以按包名控制日志级别
(5)日志可以存档,方便追踪问题
(6)可以根据配置文件调整日志,无需修改代码
JDK内置的Logging储存在java.util.logging这个包内部,这个日志系统很少使用。
import java.util.logging.Level;//导入level类
import java.util.logging.Logger;//导入logger类
public class Hello {
public static void main(String[] args) {
Logger logger = Logger.getGlobal();//调用getGlobal()方法获得logger实例
logger.info("start...");//调用info()方法输出基本信息
logger.log(Level.WARNING,"warning...");//调用log()方法输出警告级别
logger.warning("start...");//调用warning()方法输出警告信息
}
}
Java的JDK Logging设置了7个级别:
①SEVERE
②WARNING
③INFO(默认级别)
④CONFIG
⑤FINE
⑥FINER
⑦FINEST
如果我们设置了一个级别时(例如INFO),将只输出INFO和INFO以上的级别
①JVM在启动时必须读取配置文件并完成初始化
②JVM启动后无法修改配置
③需要在JVM启动时传递参数:
-Djava.util.logging.config.file=config-file-name
Commons Logging是Apache创建的日志模块:
①可以挂接不同的日志系统
②可以通过配置文件指定挂接的日志系统
③可以自动搜索并使用Log4j
④若Log4j不存在则自动使用JDK Logging(JDK >= 1.4)
public class Main {
public static void main(String[] args) {
Log log = LogFactory.getLog(Main.class);//获取Log实例,传入class
logger.info("start...");调用info()方法输出基本信息
logger.warn("end...");调用warn()方法输出警告信息
}
}
Commons Logging定义了6个日志级别:
①FATAL(非常严重的错误)
②ERROR(错误)
③WARNING(警告)
④INFO(默认)
⑤DEBUG(调试信息)
⑥TRACE(底层调试信息)
//在静态方法中引用Log:
public class Main {
static final Log log = LogFactory.getLog(Main.class);
}
//在实例方法中引用Log:
public class Person {
final Log log = LogFactory.getLog(getClass());
}
//在父类中实例化Log:
public abstract class Base {
protected final Log log = LogFactory.getLog(getClass());
}
①Commons Logging是使用最广泛的日志模块
②Commons Logging的API非常简单
③Commons Logging可以自动使用其他日志模块
Log4j是目前最流行的日志框架,拥有2个版本:
①1.x:Log4j
②2.x:Log4j2
Log4j是一个组件化设计的日志系统:
①Filter用来过滤哪些Log需要被输出而哪些Log不需要被输出
②Layout用来格式化Log信息
③在实际使用过程中不需要关心Log4j的内部API
①使用Log4j是需要把Log4j2.xml和相关jar放入classpath
②如果要更换Log4j,只需要移除Log4j.xml和相关jar
③只有扩展Log4j时,才需要引用Log4j的接口