搜索
首页Javajava教程Java关键字throw、throws、Throwable怎么用

    throw,意为“投掷、抛、扔”。Throw、Throws和Throwable三者都用于异常处理。

    1. Throwable

    Throwable在Java中是异常处理这个分支的顶级父类,其它所有异常处理的实现都依赖于Throwable

    打开Java官方文档(Java8版本),找到Throwable,它的直接子类为Error和Exception。

    Java关键字throw、throws、Throwable怎么用

    Error和Exception两者的特点在于Error异常程序无法处理,只能交由人工介入修改代码,比如栈溢出、堆溢出等等;而Exception异常可以提前发觉,并作出有效处理。

    1.1 扩展-Error

    在Error中,常见的有栈溢出和堆溢出等等。

    Java关键字throw、throws、Throwable怎么用

    举个例子,StackOverflowError

    public class ErrorTest {
        public static void main(String[] args) {
            main(args);
        }
    }

    无限递归,执行这个程序就会报栈溢出异常。

    Java关键字throw、throws、Throwable怎么用

    再比如堆异常,OutOfMemoryError

    public class ErrorTest {
        public static void main(String[] args) {
            Integer[] testArray = new Integer[1024*1024*1024];
        }
    }

    Java关键字throw、throws、Throwable怎么用

    1.2 扩展-Exception

    在Exception中就有非常多我们所熟知的异常情况了,比如NullPointerException(空指针异常)、ArrayIndexOutOfBoundsException(数组下标越界)、NumberFormatException(数字格式化异常)等等。

    public class ExceptionTest {
        public static void main(String[] args) {
            int[] testArray = null;
            System.out.println(testArray[0]);  //空指针异常
        }
    }
    public class ExceptionTest {
        public static void main(String[] args) {
            int[] testArray = new int[3];
            System.out.println(testArray[3]);   //数组下标越界
        }
    }
    public class ExceptionTest {
        public static void main(String[] args) {
            String num = "abc";
            System.out.println(Integer.parseInt(num));    //数字格式化异常
        }
    }

    2. throws

    throws应用在方法声明处,指明此方法在执行时可能会出现的异常类型。一旦该方法执行时出现异常,就会在异常代码处生成一个异常类的对象,此对象满足Throws后的异常类型时,就会被抛出。这里有两个过程,代码有异常时

    1. 生成一个异常对象;

    2. throws捕获到这个异常,将异常对象抛出

    throws和try-catch-finally一起称为异常处理的两种方式。

    try-catch-finally是在出现异常时主动处理掉异常,使得程序可以继续执行下去;而throws捕获到异常之后向上抛出异常对象,不去真正地处理这个异常。

    所谓向上抛出异常对象,是将异常对象交给调用者去处理,比如方法A调用方法B,B通过throws抛出异常,而A可以选择使用try-catch-finally处理掉异常,也可以通过throws继续向上抛出异常对象,直到异常被真正处理掉。如果一直没有方法去处理异常,异常对象最终会被抛给JVM,从而导致程序停止运行。

    @Test
    public void throwsTest(){   //调用者解决抛出的异常
        try{
            formatChange("abc");
        }
        catch (NumberFormatException e){
            System.out.println("转换格式错误!");
        }
        catch (Exception e){
            System.out.println("出现错误");
        }
    }
    private int formatChange(String str) throws NumberFormatException{    //出现异常向上抛出
        return Integer.parseInt(str);
    }

    2.1 扩展

    ——如何选择try-catch-finally还是throws?

    当一个方法中存在异常需要处理,在大多数情况下,既可以选择try-catch-finally直接处理掉这个异常,也可以选择throws向上抛出异常,交给调用者去处理(异常抛到最后,总要有一方真正地去处理这个异常,怎么处理?还是用try-catch-finally呗),在选择上比较自由,但是,出现以下两种情况时,需要遵循一定的规则(如有补充,敬请指出)。

    • 如果父类中被重写的方法没有使用throws抛出异常,则子类重写的方法也不能使用throws抛出异常,也就意味着这种情况必须使用try-catch-finally去处理。

    • 在方法A中,先后调用了另外的几种方法,这几种方法是递进关系执行的且其中很多方法都存在异常需要处理,这种情况建议被调用的几个方法使用throws向上抛出异常,在方法A中,使用try-catch-finally统一处理掉这些异常。

    针对第一条,这是一个规定,子类中重写的方法使用throws抛出的异常必须不大于父类中被重写的方法抛出异常的范围。举个例子,父类中的方法B抛出NullPointerException异常,则子类中重写B方法就不能抛出如Exception这种比NullPointerException范围更大的异常;如果父类中被重写的方法没有抛出任何异常,则子类更不能抛出异常。

    为什么?展示一段代码。

    //假设父类中的方法B抛出NullPointerException异常,子类中的方法B可以抛出Exception
    private void test(ParentClassTest parent){
        try{
            parent.B();
        }
        catch(NullPointerException e){
            System.out.println("出现了空指针异常");
        }
    }

    在本示例中,假设父类中的方法B抛出NullPointerException异常,子类中重写的方法B可以抛出Exception。那么传进给test方法的参数如果是父类的实例化对象,那么调用test方法没有任何问题。如果传进的参数是子类的实例化对象,再去调用子类重写的方法B,那么就有可能抛出Exception异常,try-catch结构就压不住这个异常了,这显然是一个不合理的操作。

    针对第二条,假设方法A中调用了方法C、D、E,这三个方法都有可能产生异常,且存在递进关系,也就是D、E执行需要C执行完成、E执行依赖C、D执行完成。那么就推荐在C、D、E中向上抛出异常,在方法A中集中处理。为什么?如果C、D、E都是向上抛出异常,而A使用try-catch-finally去处理这个异常,如果某个方法真的出现异常,则不再继续执行。而如果C、D、E都使用try-catch-finally直接解决掉异常,那么即使产生了异常,方法A也不会接收到异常的产生,那么还会接着往下执行,但是C出现了异常,再执行D、E没有任何意义。

    3. throw

    如果在程序编写时有手动抛出异常的需求,则可以使用throw

    throw使用在方法体内。与try-catch-finally和throws都不同,异常处理的两个阶段:1.遇到异常,生成异常对象;2.捕获到异常,进行抛出或处理。try-catch-finally和throws都处在第二个阶段,都是捕获到异常后的相关处理,一般使用系统根据异常类型自动生成的异常对象进行处理。而throw应用在第一阶段,手动地产生一个异常对象。

    举一个例子,判断一个数值是否为非负数,如果为负数,则抛出异常。

    class ThrowTest{
        private int Number;
        public void judge(int num){
            if(num>=0){
                this.Number = num;
            }
            else{
                throw new RuntimeException("传入参数为负数");
            }
        }
    }
    @Test
    public void test2(){
        ThrowTest throwTest = new ThrowTest();
        throwTest.judge(-100);
    }

    成功抛出异常。

    Java关键字throw、throws、Throwable怎么用

    使用try-catch捕获一下异常。

    @Test
    public void test2(){
        ThrowTest throwTest = new ThrowTest();
        try{
            throwTest.judge(-100);
        }
        catch (RuntimeException e){
            System.out.println(e.getMessage());
        }
    }

    Java关键字throw、throws、Throwable怎么用

    如果把throw抛出的异常改为Exception,则直接报错,也就是不能编译。Exception包含两种异常:编译时异常和运行时异常,前者在编译前就要检查是否有可能产生编译时异常;后者是在编译后运行时才会判断的异常。而throw new Exception包含了编译时异常,需要显式处理掉这个异常,怎么处理?try-catch-finally或者throws

    Java关键字throw、throws、Throwable怎么用

    class ThrowTest{
        private int Number;
        public void judge(int num) throws Exception{
            if(num>=0){
                this.Number = num;
            }
            else{
                throw new Exception("传入参数为负数");
            }
        }
    }

    调用方也要随着进行更改。

    @Test
    public void test2(){
        ThrowTest throwTest = new ThrowTest();
        try{
            throwTest.judge(-100);
        }
        catch (RuntimeException e){
            System.out.println(e.getMessage());
        }
        catch (Exception e){
            System.out.println(e.getMessage());
        }
    }

    Java关键字throw、throws、Throwable怎么用

    3.1 扩展

    ——自定义异常类

    throw还可以抛出自定义异常类。

    自定义异常类的声明需要继承于现有的异常体系。

    class MyException extends RuntimeException{
        static final long serialVersionUID = -703489719076939L;   //可以认为是一种标识
        public MyException(){}
        public MyException(String message){
            super(message);
        }
    }

    此时我们可以抛出自定义的异常

    class ThrowTest{
        private int Number;
        public void judge(int num) throws MyException{
            if(num>=0){
                this.Number = num;
            }
            else{
                throw new MyException("不能输入负数");
            }
        }
    }

    调用者修改

    @Test
    public void test2(){
        ThrowTest throwTest = new ThrowTest();
        try{
            throwTest.judge(-100);
        }
        catch (MyException e){
            System.out.println(e.getMessage());
        }
    }

    Java关键字throw、throws、Throwable怎么用

    以上是Java关键字throw、throws、Throwable怎么用的详细内容。更多信息请关注PHP中文网其他相关文章!

    声明
    本文转载于:亿速云。如有侵权,请联系admin@php.cn删除
    平台独立性如何使企业级的Java应用程序受益?平台独立性如何使企业级的Java应用程序受益?May 03, 2025 am 12:23 AM

    Java在企业级应用中被广泛使用是因为其平台独立性。1)平台独立性通过Java虚拟机(JVM)实现,使代码可在任何支持Java的平台上运行。2)它简化了跨平台部署和开发流程,提供了更大的灵活性和扩展性。3)然而,需注意性能差异和第三方库兼容性,并采用最佳实践如使用纯Java代码和跨平台测试。

    考虑到平台独立性,Java在物联网(物联网)设备的开发中扮演什么角色?考虑到平台独立性,Java在物联网(物联网)设备的开发中扮演什么角色?May 03, 2025 am 12:22 AM

    JavaplaysigantroleiniotduetoitsplatFormentence.1)itallowscodeTobewrittenOnCeandrunonVariousDevices.2)Java'secosystemprovidesuseusefidesusefidesulylibrariesforiot.3)

    描述一个方案,您在Java中遇到了一个特定于平台的问题以及如何解决。描述一个方案,您在Java中遇到了一个特定于平台的问题以及如何解决。May 03, 2025 am 12:21 AM

    ThesolutiontohandlefilepathsacrossWindowsandLinuxinJavaistousePaths.get()fromthejava.nio.filepackage.1)UsePaths.get()withSystem.getProperty("user.dir")andtherelativepathtoconstructthefilepath.2)ConverttheresultingPathobjecttoaFileobjectifne

    Java平台独立对开发人员有什么好处?Java平台独立对开发人员有什么好处?May 03, 2025 am 12:15 AM

    Java'splatFormIndenceistificantBecapeitAllowSitallowsDevelostWriTecoDeonCeandRunitonAnyPlatFormwithAjvm.this“ writeonce,runanywhere”(era)橱柜橱柜:1)交叉plat formcomplibility cross-platformcombiblesible,enablingDeploymentMentMentMentMentAcrAptAprospOspOspOssCrossDifferentoSswithOssuse; 2)

    将Java用于需要在不同服务器上运行的Web应用程序的优点是什么?将Java用于需要在不同服务器上运行的Web应用程序的优点是什么?May 03, 2025 am 12:13 AM

    Java适合开发跨服务器web应用。1)Java的“一次编写,到处运行”哲学使其代码可在任何支持JVM的平台上运行。2)Java拥有丰富的生态系统,包括Spring和Hibernate等工具,简化开发过程。3)Java在性能和安全性方面表现出色,提供高效的内存管理和强大的安全保障。

    JVM如何促进Java的'写作一次,在任何地方运行”(WORA)功能?JVM如何促进Java的'写作一次,在任何地方运行”(WORA)功能?May 02, 2025 am 12:25 AM

    JVM通过字节码解释、平台无关的API和动态类加载实现Java的WORA特性:1.字节码被解释为机器码,确保跨平台运行;2.标准API抽象操作系统差异;3.类在运行时动态加载,保证一致性。

    Java的较新版本如何解决平台特定问题?Java的较新版本如何解决平台特定问题?May 02, 2025 am 12:18 AM

    Java的最新版本通过JVM优化、标准库改进和第三方库支持有效解决平台特定问题。1)JVM优化,如Java11的ZGC提升了垃圾回收性能。2)标准库改进,如Java9的模块系统减少平台相关问题。3)第三方库提供平台优化版本,如OpenCV。

    说明JVM执行的字节码验证的过程。说明JVM执行的字节码验证的过程。May 02, 2025 am 12:18 AM

    JVM的字节码验证过程包括四个关键步骤:1)检查类文件格式是否符合规范,2)验证字节码指令的有效性和正确性,3)进行数据流分析确保类型安全,4)平衡验证的彻底性与性能。通过这些步骤,JVM确保只有安全、正确的字节码被执行,从而保护程序的完整性和安全性。

    See all articles

    热AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智能驱动的应用程序,用于创建逼真的裸体照片

    AI Clothes Remover

    AI Clothes Remover

    用于从照片中去除衣服的在线人工智能工具。

    Undress AI Tool

    Undress AI Tool

    免费脱衣服图片

    Clothoff.io

    Clothoff.io

    AI脱衣机

    Video Face Swap

    Video Face Swap

    使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

    热工具

    SublimeText3汉化版

    SublimeText3汉化版

    中文版,非常好用

    螳螂BT

    螳螂BT

    Mantis是一个易于部署的基于Web的缺陷跟踪工具,用于帮助产品缺陷跟踪。它需要PHP、MySQL和一个Web服务器。请查看我们的演示和托管服务。

    MinGW - 适用于 Windows 的极简 GNU

    MinGW - 适用于 Windows 的极简 GNU

    这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。

    mPDF

    mPDF

    mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),

    Atom编辑器mac版下载

    Atom编辑器mac版下载

    最流行的的开源编辑器