Java异常

简介: Java异常

Java异常

语法错误,Error和Exception

语法错误

// 语法错误
String s = "123";
// Integer i = (Integer) s;// 语法错误
Integer i = Integer.parseInt(s);//应该使用Integer的静态方法进行类型转换

错误Error

test();// 递归没有跳出逻辑 java.lang.StackOverflowError 栈溢出

public static void test() {
    test();
}

程序执行时的一个错误场景,在Java中,提供一种特殊类:Error,来描述这种场景。

一旦发生会程序会自动停止,无法恢复。应该尽量避免。

异常Exception

异常(相对于正常逻辑),在java中使用Exception来表示异常

User user = null;
System.out.println(user.toString());// 空指针异常 java.lang.NullPointerException

异常分为两种:

  • 可通过代码来恢复正常逻辑执行的异常,称之为运行时异常:RuntimeException
  • 不可通过代码来恢复正常逻辑执行的异常,称之为编译器异常(在编译时进行提醒):Exception

对于编译器异常,并不是会真正的抛出异常,而是在编译时进行提醒。比如:程序连接服务器,但是服务器没开,这时候编译器会提醒你,针对这种情况,必须启动服务器。否则就会发生错误。

对于异常,会根据异常的范围确定一个从属关系。在Java中,这种从属关系表现为继承类,范围越小,异常的处理信息越详细,这种异常被称为子类异常。范围越大,越粗略,被称为父类异常。

其中,它的最大范围的类是Exception类。

public class RuntimeException extends Exception{...}

try/catch捕获异常

假如我们有下面一段代码:

int i = 10;
int j = 0;
System.out.println(i / j);// Exception in thread "main" java.lang.ArithmeticException: / by zero

在上面这段代码中,程序会因为一个异常而中断运行,而这个异常产生的原因是:除法运算的的被除数为零。这样的代码十分脆弱。

为了提升代码的健壮性,Java为我们提供了try/catch来捕获异常并在出现该异常时来通过指定的代码来解决异常,从而使程序不会中断

int i = 10;
int j = 0;
try {
    System.out.println(i / j);
} catch (ArithmeticException e) {
    System.out.println("发生异常的原因是:" + e.getCause());//发生异常的原因是:null 返回null因为:原因不存在或未知
    System.out.println("异常信息为:" + e.getMessage());//异常信息为:/ by zero
    j = 10;
} finally {
    System.out.println("最终执行的代码");
}
System.out.println(i / j);//1

我们对可能产生异常的部分进行了处理,所以说,程序就不会中断了。

对于try/catch的语法:

try {
    // 可能发生异常的代码
    // 如果出现异常,那么JVM会将异常进行封装,形成一个具体的异常类,然后将这个异常抛出
} catch (异常类的实例对象) {
    // 捕获异常后,异常的解决方案
} catch(通常情况下异常捕获的顺序为范围从小到大){
    //...
}finally {
    System.out.println("最终执行的代码");
}

通常情况下异常捕获的顺序为范围从小到大,在捕获到异常后就不会捕获其他异常。在执行完catch中的语句后会执行finally中的语句。finally不是必须的。

所有的异常都可以被抛出,因为Exception继承了Throwable。

public class Exception extends Throwable{...}

常见异常

RuntimeException可以不使用try/catch,但是对于异常我们应该提前发现,提前解决

举个例子:

int i = 10;
int j = 0;
if (j != 0) {
    System.out.println(i / j);
}

TheUser user = null;
if (user != null) {
    System.out.println(user.toString());
}

空指针异常和算术异常

上面两个异常都继承自RuntimeException

public class NullPointerException extends RuntimeException{...}
public class ArithmeticException extends RuntimeException{...}

上面的例子中,我们没有写try/catch,看上去我们的代码简洁了。但是,在逻辑上我们不应该判断j是否为0来执行语句,因为本来被除数就不应该为零。我们也不应该通过对象是否为空(null)来判断是否执行语句。

我们应该提前发现提前解决

TheUser user = null;
try {
    System.out.println(user.toString());

} catch (NullPointerException e) {
    user = new TheUser();
} finally {
    System.out.println(user.toString());//chapter06.TheUser@4eec7777
}

但如果我们在TheUser中声明静态属性,并且初始化值。这时候将user指向null是可以取到name的值的。因为name是静态方法,与实例对象无关。

public class JavaException_03 {
    public static void main(String[] args) {
        TheUser user = null;
        try {
            System.out.println(user.name);

        } catch (NullPointerException e) {
            System.out.println("对象为空的时候需要分析数据为空的原因");
        }
        // kevin
    }
}

class TheUser {
    public static String name = "kevin";
}

当调用了一个为空(null)对象的成员属性和方法时会发生异常,但是静态的属性和方法不受影响,因为静态属性方法与实例对象无关。

索引越界

当下标大于等于数组长度时会出现索引越界

int[] ints = {1, 2, 3};
System.out.println(ints[3]);//Index 3 out of bounds for length 3

数组的异常类是ArrayIndexOutOfBoundsException

public class ArrayIndexOutOfBoundsException extends IndexOutOfBoundsException{...}
String strings = "abc";
System.out.println(strings.charAt(3));

字符串的异常类是StringIndexOutOfBoundsException

public class StringIndexOutOfBoundsException extends IndexOutOfBoundsException{...}

但是注意substring方法的参数最大值是字符串长度而不是下标。

String strings = "abc";
System.out.println(strings.substring(3));
System.out.println(strings.substring(4));// begin 4, end 3, length 3

格式化异常和类型转换错误

String s = "a123";
int i = Integer.parseInt(s);// java.lang.NumberFormatException: For input string: "a123"

我们知道RuntimeException可以使用代码来恢复,我们可以使用正则表达式来对字符串进行处理:

String s = "a123b456";
String regex = "[^0-9]";
s = s.replaceAll(regex, "").trim();
int i = Integer.parseInt(s);
System.out.println(i);// 123456

当两个无关的类进行转换的时候会报:ClassCastException

public class JavaException_05 {
    public static void main(String[] args) {
        Object obj = new Obj();
        TheObj theObj = (TheObj) obj;// java.lang.ClassCastException
    }
}

class Obj {}
class TheObj {}

我们可以这样解决:

Object obj = new Obj();
if (obj instanceof TheObj) {
    TheObj theObj = (TheObj) obj;// java.lang.ClassCastException
}

异常转换

public class JavaException_06 {
    public static void main(String[] args) {
        Util.test(10, 0);// java.lang.ArithmeticException: / by zero
    }
}

class Util {
    public static void test(int i, int j) {
        System.out.println(i / j);
    }
}

如果方法中可能出现问题,我们应该提前声明,告诉其他人,我的方法可能会出现异常

public static void test(int i, int j) throws ArithmeticException {
    System.out.println(i / j);
}

也可以在调用方法的地方进行抛出异常。

public static void main(String[] args) throws Exception {
    Util.test(10, 0);// java.lang.ArithmeticException: / by zero
}

但是像上面的写法不抛出异常也不会报错,因为我们当前的异常是RuntimeException(运行时异常),编译器不会提示。

但是我们希望调用方法时能够手动抛出异常,这时我们不能抛出RuntimeException了。

如果程序中需要手动抛出异常对象,需要使用throw关键字,然后new出异常对象。(注意new出的对象类型与抛出的要一致)

public class JavaException_06 {
    public static void main(String[] args) throws Exception {
        Util.test(10, 0);// java.lang.Exception
    }
}

class Util {
    public static void test(int i, int j) throws Exception {
        try {
            System.out.println(i / j);
        } catch (ArithmeticException e) {
            throw new Exception();
        }
    }
}

而这时候我们抛出的异常也不是ArithmeticException,而是手动抛出的Exception。

自定义异常

我们之前使用的都是java预先定义好的异常,有时候并不能准确描述出我们异常的场景,比如我们业务中的场景。

如果要准确描述,需要我们自己定义异常,来描述这个场景:

package chapter06;

public class JavaException_07 {
    public static void main(String[] args) {
        try {
            login("kevin", "kevin666");
        } catch (AccountException e) {
            System.out.println("账户错误");
        } catch (PasswordException e) {
            System.out.println("密码异常");
        } catch (LoginException e) {
            System.out.println("其他登录异常");
        }
        // 账户错误
    }

    public static void login(String account, String password) {
        if (!"admin".equals(account)) {
            throw new AccountException("账户不正确");
        }
        if (!"admin".equals(password)) {
            throw new PasswordException("密码不正确");
        }
        System.out.println("登录成功");
    }
}

class AccountException extends LoginException {
    AccountException(String message) {
        super(message);
    }
}

class PasswordException extends LoginException {
    PasswordException(String message) {
        super(message);
    }
}


class LoginException extends RuntimeException {
    LoginException(String message) {
        super(message);
    }
}

如果我们继承Exception而不是RuntimeException,我们需要在抛出异常的方法签名出也抛出异常,如果有多种可能,使用逗号,隔开:

package chapter06;

public class JavaException_07 {
    public static void main(String[] args) {
//...
   }

  public static void login(String account, String password) throws AccountException, PasswordException {
        if (!"admin".equals(account)) {
            throw new AccountException("账户不正确");
        }
        if (!"admin".equals(password)) {
            throw new PasswordException("密码不正确");
        }
        System.out.println("登录成功");
}

class AccountException extends LoginException {
//...
}

class PasswordException extends LoginException {
//...
}
 
class LoginException extends Exception {
//...
}

也可以直接抛出LoginException

相关文章
|
2月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
90 1
|
2月前
|
Java API 调度
如何避免 Java 中的 TimeoutException 异常
在Java中,`TimeoutException`通常发生在执行操作超过预设时间时。要避免此异常,可以优化代码逻辑,减少不必要的等待;合理设置超时时间,确保其足够完成正常操作;使用异步处理或线程池管理任务,提高程序响应性。
129 12
|
2月前
|
Java
在 Java 中,如何自定义`NumberFormatException`异常
在Java中,自定义`NumberFormatException`异常可以通过继承`IllegalArgumentException`类并重写其构造方法来实现。自定义异常类可以添加额外的错误信息或行为,以便更精确地处理特定的数字格式转换错误。
49 1
|
2月前
|
IDE 前端开发 Java
怎样避免 Java 中的 NoSuchFieldError 异常
在Java中避免NoSuchFieldError异常的关键在于确保类路径下没有不同版本的类文件冲突,避免反射时使用不存在的字段,以及确保所有依赖库版本兼容。编译和运行时使用的类版本应保持一致。
101 7
|
2月前
|
Java 编译器
如何避免在 Java 中出现 NoSuchElementException 异常
在Java中,`NoSuchElementException`通常发生在使用迭代器、枚举或流等遍历集合时,尝试访问不存在的元素。为了避免该异常,可以在访问前检查是否有下一个元素(如使用`hasNext()`方法),或者使用`Optional`类处理可能为空的情况。正确管理集合边界和条件判断是关键。
116 6
|
2月前
|
Java
Java异常捕捉处理和错误处理
Java异常捕捉处理和错误处理
76 1
|
2月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
98 2
|
2月前
|
Java
如何在 Java 中处理“Broken Pipe”异常
在Java中处理“Broken Pipe”异常,通常发生在网络通信中,如Socket编程时。该异常表示写入操作的另一端已关闭连接。解决方法包括:检查网络连接、设置超时、使用try-catch捕获异常并进行重试或关闭资源。
155 5
|
2月前
|
存储 安全 Java
如何避免 Java 中的“ArrayStoreException”异常
在Java中,ArrayStoreException异常通常发生在尝试将不兼容的对象存储到泛型数组中时。为了避免这种异常,确保在操作数组时遵循以下几点:1. 使用泛型确保类型安全;2. 避免生类型(raw types)的使用;3. 在添加元素前进行类型检查。通过这些方法,可以有效防止 ArrayStoreException 的发生。
59 3
|
3月前
|
人工智能 Oracle Java
解决 Java 打印日志吞异常堆栈的问题
前几天有同学找我查一个空指针问题,Java 打印日志时,异常堆栈信息被吞了,导致定位不到出问题的地方。
59 2