c web网站开发 pdf/30个免费货源网站
还有大家来到权权的博客~欢迎大家对我的博客提出意见哦,有错误会及时改进的~
点击进入我的博客主页
目录
- 一、异常的概念及体系结构
- 1.1 异常的概念
- 1.2 异常的体系结构
- 1.3异常的分类
- 二、异常的处理
- 2.1防御式编程
- 2.2 异常的抛出
- 2.3 异常的捕获
- 2.3.1异常声明throws
- 2.3.2 try-catch捕获并处理
- 2.3.3 finally
- 2.4 异常的处理流程
- 三、自定义异常类
一、异常的概念及体系结构
1.1 异常的概念
在生活中,一个人表情痛苦,出于关心,可能会问:你是不是生病了,需要我陪你去看医生吗?
在程序中也是一样,程序猿是一帮办事严谨、追求完美的高科技人才。在日常开发中,绞尽脑汁将代码写的尽善尽美,在程序运行过程中,难免会出现一些奇奇怪怪的问题。有时通过代码很难去控制,比如:数据格式不对、网络不通畅、内存报警等。
在Java中,将程序执行过程中发生的不正常行为称为异常。比如之前写代码时经常遇到的:
1.算术异常:
2.数组越界异常:
3.空指针异常:
从上述过程中可以看到,java中不同类型的异常,都有与其对应的类来进行描述。
1.2 异常的体系结构
异常种类繁多,为了对不同异常或者错误进行很好的分类管理,Java内部维护了一个异常的体系结构:
看上图:
- Throwable:是异常体系的顶层类,其派生出两个重要的子类, Error和Exception
- Error:指的是Java虚拟机无法解决的严重问题,比如:JVM的内部错误、资源耗尽等,典型代表;
- Exception:异常产生后程序员可以通过代码进行处理,使程序继续执行。
1.3异常的分类
异常可能在编译时发生,也可能在程序运行时发生,根据发生的时机不同,可以将异常分为:
1.编译时异常:
2.运行时异常:
RunTimeException以及其子类对应的异常,都称为运行时异常。比如:NullPointerException、ArrayIndexOutOfBoundsException、ArithmeticException。
注意:
编译时出现的语法性错误,不能称之为异常。例如将 System.out.println 拼写错了, 写成了system.out.println. 此时编译过程中就会出错, 这是 “编译期” 出错。而运行时指的是程序已经编译通过得到class 文件了, 再由 JVM 执行过程中出现的错误。
二、异常的处理
在Java中,处理异常主要有这五个关键字:thorw 抛出异常,try 里面放可能出现的异常代码,catch 捕捉异常,finally 必须执行的特定代码,thorws 声明异常。
2.1防御式编程
事前防御型:
public class test {public static void main(String[] args) {boolean bool=true;if(bool){System.out.println("进入游戏失败");}if(!bool){System.out.println("进入游戏成功");}}
}
缺陷:正常流程和错误处理流程代码混在一起, 代码整体显的比较混乱.
事后认错型:
public class test {public static void main(String[] args) {int[] num=new int[10];try {for (int i = 0; i <=num.length; i++) {System.out.println(num[i]);}}catch (ArrayIndexOutOfBoundsException e){System.out.println("捕获ArrayIndexOutOfBoundsException...");}finally {System.out.println("finally...");}}
}
优势:正常流程和错误流程是分离开的, 程序员更关注正常流程,代码更清晰,容易理解代码。
2.2 异常的抛出
在编写程序时,如果程序中出现错误,此时就需要将错误的信息告知给调用者,比如:参数检测。
在Java中,可以借助throw关键字,抛出一个指定的异常对象,将错误信息告知给调用者。具体语法如下:
throw new XXXException("异常产生的原因")
【需求】:实现一个获取数组中任意位置元素的方法。
public class test {public static int getArrayValue (int[] array, int index) {if (array == null) {throw new NullPointerException("getArrayValue::传递的数组为空");}if (index < 0 || index >= array.length) {throw new ArrayIndexOutOfBoundsException("getArrayValue::数组下标越界了!");}System.out.println("测试抛出异常后会不会执行后续代码!");return array[index];}public static void main(String[] args) {int[] array = {1,2,3,4,5};int ret = getArrayValue(array, 10);System.out.println(ret);}
}
【注意事项】
- throw必须写在方法体内部
- 抛出的对象必须是Exception 或者 Exception 的子类对象
- 如果抛出的是 RunTimeException 或者 RunTimeException 的子类,则可以不用处理,直接交给JVM来处理4. 如果抛出的是编译时异常,用户必须处理,否则无法通过编译
- 异常一旦抛出,其后的代码就不会执行
2.3 异常的捕获
异常的捕获,也就是异常的具体处理方式,主要有两种:异常声明throws 以及 try-catch捕获处理。
2.3.1异常声明throws
处在方法声明时参数列表之后,当方法中抛出编译时异常,用户不想处理该异常,此时就可以借助throws将异常抛给方法的调用者来处理。即当前方法不处理异常,提醒方法的调用者处理异常。
语法格式:
修饰符 返回值类型 方法名(参数列表) throws 异常类型1,异常类型2...{}
假设我们有一个方法,里面可能会发生异常,但是我们不想处理这个异常,那么就可以声明这个异常,那交给谁处理呢?谁调用了这个方法谁处理,假设是 main 方法调用了,就由 main 方法处理,如果 main 方法也不想处理,也可以使用 throws 声明这个异常,即交给JVM处理,如何自己处理异常,后面的 try - catch 就会讲到!
注意:1. throws必须跟在方法的参数列表之后
2. 声明的异常必须是 Exception 或者 Exception 的子类
3. 方法内部如果抛出了多个异常,throws之后必须跟多个异常类型,之间用逗号隔开,如果抛出多个异常类型具有父子关系,直接声明父类即可。
4. 调用声明抛出异常的方法时,调用者必须对该异常进行处理,或者继续使用throws抛。
5.将光标放在抛出异常方法上,alt + Insert 快速处理(idea)
2.3.2 try-catch捕获并处理
语法格式:
try{//放置可能发生异常的代码}catch (Exception e){//对异常进行不糊}catch (Exception e){//对异常进行不糊}finally {//这里的代码一定会被执行}
关于异常的处理方式:
1.异常的种类有很多, 我们要根据不同的业务场景来决定.
2.对于比较严重的问题(例如和算钱相关的场景), 应该让程序直接崩溃, 防止造成更严重的后果.
3.对于不太严重的问题(大多数场景), 可以记录错误日志, 并通过监控报警程序及时通知程序猿
4.对于可能会恢复的问题(和网络相关的场景), 可以尝试进行重试
5.在我们当前的代码中采取的是经过简化的第二种方式. 我们记录的错误日志是出现异常的方法调用信息, 能很快速的让我们找到出现异常的位置. 以后在实际工作中我们会采取更完备的方式来记录异常信息.
【注意事项】
1. try块内抛出异常位置之后的代码将不会被执行
2. 如果抛出异常类型与catch时异常类型不匹配,即异常不会被成功捕获,也就不会被处理,继续往外抛,直到JVM收到后中断程序----异常是按照类型来捕获的.
3. try中可能会抛出多个不同的异常对象,则必须用多个catch来捕获----即多种异常,多次捕获.
public class test {public static void main(String[] args) {try{int[] num=null; //这里发生的是空指针异常String[] str=new String[10];str[11]="hello";}catch (ArrayIndexOutOfBoundsException e) {//此处捕捉的是数组越界异常System.out.println("捕获了ArrayIndexOutOfBoundsException。。。异常");e.printStackTrace();}catch (NullPointerException e){System.out.println("捕获了NullPointerException。。");e.printStackTrace();}finally {System.out.println("finally...执行了");}}
}
4.如果多个异常的处理方式是完全相同, 也可以写成这样:
catch(ArrayIndexOutOfBoundsException|NullPointerExceptione){
}
5.不会同时抛出多个异常
6.如果异常之间具有父子关系,一定是子类异常在前catch,父类异常在后catch,否则语法错误:
7.可以通过一个catch捕获所有的异常,即多个异常,一次捕获(不推荐)
public class test {public static void main(String[] args) {int[] num={1,2,3};try{System.out.println(num[4]);}catch (Exception e){e.printStackTrace();}}
}
2.3.3 finally
在写程序时,有些特定的代码,不论程序是否发生异常,都需要执行,比如程序中打开的资源:网络连接、数据库连接、IO流等,在程序正常或者异常退出时,必须要对资源进进行回收。另外,因为异常会引发程序的跳转,可能导致有些语句执行不到,finally就是用来解决这个问题的。
语法:
try{//放置可能发生异常的代码}catch (Exception e){//对异常进行不糊}catch (Exception e){//对异常进行不糊}finally {//这里的代码一定会被执行}//...code 如果没有抛出异常,或者异常被捕获处理了,这里的代码也会执行
问题:既然 finally 和 try-catch-finally 后的代码都会执行,那为什么还要有finally呢?
这里我们用一个例子来看一下,实现一个 getData 方法,方法内部输入一个数字,并返回这个数字:
public static int getData() {Scanner scanner = null;try {scanner = new Scanner(System.in);int data = scanner.nextInt();return data;} catch (InputMismatchException ex) {ex.printStackTrace();} finally {System.out.println("finally中代码");}System.out.println("try-catch-finally之后代码");if (null != scanner) {scanner.close();}return 0;public static void main(String[] args) {intdata=getData();System.out.println(data);}
}
那这里我们有一个问题,如果你是正常输入的,那么将会直接返回 data,并不会走到后面的代码,也就是不会释放输入流,造成资源浪费,所以我们可以把释放输入流语句放到 finally 中,因为 不管是否抛出异常,是否被捕获,finally 中的语句都会被执行。
finally 中的代码一定会执行的,一般在 finally 中进行一些资源清理的扫尾工作。
public static int getData() {Scanner scanner = null;try {scanner = new Scanner(System.in);int data = scanner.nextInt();return data;} catch (InputMismatchException ex) {ex.printStackTrace();} finally {System.out.println("finally中代码");scanner.close();//收尾}System.out.println("try-catch-finally之后代码");return 0;public static void main(String[] args) {intdata=getData();System.out.println(data);}
}
2.4 异常的处理流程
先是执行 try 中的代码,如果里面的代码发生了异常,就会立刻结束 try 中的代码,不会执行 try 中后续代码,接着会去看 catch 中的异常类型与 try 中发生异常的类型能否匹配,如果找到了匹配的异常类型,则会执行对应 catch 中的代码,如果没有找到匹配的异常类型,就会将异常向上传递到上层调用者,无论是否发生异常,是否匹配,finally 中的代码都会在方法结束前执行!如果上层调用者也没有处理这个异常,就继续向上传递,一直到 main 方法也没有合适的代码处理,就会交给 JVM 来进行处理,这个时候,程序就会异常终止!
三、自定义异常类
Java 中虽然已经内置了丰富的异常类, 但是并不能完全表示实际开发中所遇到的一些异常,此时就需要维护符合我们实际情况的异常结构。
例如, 我们实现一个用户登陆功能.
class login{private String userName="root";private String password="1234";public String getUserName() {return userName;}public void setUserName(String userName) {this.userName = userName;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}public void logInfo(String userName, String password) {if (!this.userName.equals(userName)) {System.out.println("用户名错误");return;}if (!this.password.equals(password)) {System.out.println("密码错误");return;}System.out.println("登陆成功");}
}
public class test {public static void main(String[] args) {login login=new login();login.logInfo("ABC","123");}
}
怎么用异常的方式去写呢?
1.自定义一个类,然后继承 Exception 或者 RuntimeException 类.
2.实现一个带有 String 类型参数的构造方法,参数也就是出现异常的原因.
我们先看一下其他异常是怎么实现的:
这里以ArithmeticException是怎么实现的为例子:
然后创建一个UserNameException类然后模仿上面的让它继承RuntimeException
然后修改一下代码改成异常的方式:
login类代码:
class login{private String userName="root";private String password="1234";public String getUserName() {return userName;}public void setUserName(String userName) {this.userName = userName;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}public void logInfo(String userName, String password) {if (!this.userName.equals(userName)) {throw new UserNameException("用户名错误");}if (!this.password.equals(password)) {throw new UserNameException("密码错误");}System.out.println("登陆成功");}
}
public class test {public static void main(String[] args) {login login=new login();login.logInfo("ABC","123");}
}
自定义异常类代码:
public class UserNameException extends RuntimeException{public UserNameException() {super();}public UserNameException(String s) {super(s);}
}
测试代码:
public class test {public static void main(String[] args) {login login=new login();try{login.logInfo("ABC","123");}catch (UserNameException e){System.out.println("捕获了UserNameException...");e.printStackTrace();}finally {System.out.println("finally,,,");}}
}
运行结果:
可以看到自定义异常成功。
注意:
如果自定义异常类继承了 Exception类,默认是受检查异常(编译时异常)。
如果自定义异常类继承了 RuntimeException,默认是不受检查异常(运行时异常。
欧耶!!!我学会啦!!!!
相关文章:

java---认识异常(详解)
还有大家来到权权的博客~欢迎大家对我的博客提出意见哦,有错误会及时改进的~点击进入我的博客主页 目录 一、异常的概念及体系结构1.1 异常的概念1.2 异常的体系结构1.3异常的分类 二、异常的处理2.1防御式编程2.2 异常的抛出2.3 异常的捕获2.3.1异常声明throws2.3.…...

Linux基础学习笔记
Linux基础学习笔记 Linux目录结构: 具体的目录结构: /bin [重点] (/usr/bin 、 /usr/local/bin) • 是Binary的缩写, 这个目录存放着最经常使用的命令 /home [重点] • 存放普通用户的主目录,在Linux中每个用户都有一个自己的目录,一…...

自动泊车端到端算法 ParkingE2E 介绍
01 算法介绍 自主泊车是智能驾驶领域中的一项关键任务。传统的泊车算法通常使用基于规则的方案来实现。因为算法设计复杂,这些方法在复杂泊车场景中的有效性较低。 相比之下,基于神经网络的方法往往比基于规则的方法更加直观和多功能。通过收集大量专家…...

《手写Spring渐进式源码实践》实践笔记(第十七章 数据类型转换)
文章目录 第十七章 数据类型转换工厂设计实现背景技术背景Spring数据转换实现方式类型转换器(Converter)接口设计实现 业务背景 目标设计实现代码结构类图实现步骤 测试事先准备属性配置文件转换器工厂Bean测试用例测试结果: 总结 第十七章 数…...

W3C HTML 活动
关于W3C(万维网联盟)的HTML活动,我们可以从HTML的不同版本的发展历程中了解其主要的活跃时期和贡献。 HTML 2.0:这个版本的HTML是由Internet工程工作小组(IETF)的HTML工作组于1996年开发的。它是HTML的早期…...

机器学习—为什么我们需要激活函数
如果我们使用神经网络中每个神经元的线性激活函数,回想一下这个需求预测示例,如果对所有节点使用线性激活函数,在这个神经网络中,事实证明,这个大神经网络将变得与线性回归没有什么不同,所以这将挫败使用神…...

软考系统架构设计师论文:论软件的可靠性评价
试题四 论软件的可靠性评价 软件可靠性评价是软件可靠性活动的重要组成部分,既适用于软件开发过程,也可针对最 终软件系统。在软件开发过程中使用软件可靠性评价,可以使用软件可靠性模型,估计软件当前的可靠性,以确认是否可以终止测试并发布软件,同时还可以预计软件要达…...

C++:线程(thread)的创建、调用及销毁
在 C 中,线程的管理主要依赖于标准库 std::thread,自 C11 起,这一功能被标准化,使得我们能够更加方便地创建、管理和销毁线程。这里我们详细讲解线程的创建、调用和销毁流程。 1. 线程的创建 创建线程通常是为了在单独的线程中执…...

关于随身wifi,看了再决定要不要买!2024年最受欢迎的随身wifi品牌推荐!
话费、流量费缴纳起来肉疼,毕竟不是每个月都有很大需求,主打一个该省省该花花。特别是短租人群、在校学生、出差或旅游的人群、追求高性价比的人群,随身Wifi特别实用,出门当WiFi,在家当宽带,两不耽误&#…...

SpringMVC总结 我的学习笔记
SpringMVC总结 我的学习笔记 一、SpringMVC简介1.MVC2.SpringMVC概述3. SpringMVC中的核心组件4.SpringMVC核心架构流程 二、SpringMVC框架实例具体实现使用注解实现 四、数据处理及跳转1.结果跳转方式2.处理器方法的参数与返回值处理提交数据数据显示到前端 五、RestFul风格1.…...

DevCheck Pro手机硬件检测工具v5.33
前言 DevCheck Pro是一款手机硬件和操作系统信息检测查看工具,该软件的功能非常强大,为用户提供了系统、硬件、应用程序、相机、网络、电池等一系列信息查看功能 安装环境 [名称]:DevCheckPro [版本]:5.33 [大小]&a…...

数据分析ReAct工作流
让我用一个数据分析项目的例子来展示plan-and-execute框架的应用。这个例子会涉及数据处理、分析和可视化等任务。 from typing import List, Dict, Any from dataclasses import dataclass import json from enum import Enum import logging from datetime import datetime#…...

Rust-AOP编程实战
文章本天成,妙手偶得之。粹然无疵瑕,岂复须人为?君看古彝器,巧拙两无施。汉最近先秦,固已殊淳漓。胡部何为者,豪竹杂哀丝。后夔不复作,千载谁与期? ——《文章》宋陆游 【哲理】文章…...

Flutter鸿蒙next 中的 Expanded 和 Flexible 使用技巧详解
在 Flutter 开发中,Expanded 和 Flexible 是两个非常常用的布局控件,它们可以帮助开发者更加灵活地管理 UI 布局的空间分配。虽然它们看起来非常相似,但它们的功能和使用场景有所不同。理解这两者的区别,能帮助你在构建复杂 UI 布…...

【微信小游戏学习心得】
这里是引用 微信小游戏学习心得 简介了解微信小游戏理解2d游戏原理数据驱动视图总结 简介 本人通过学习了解微信小游戏,学习微信小游戏,加深了对前端框架,vue和react基于数据驱动视图的理解,及浏览器文档模型和javaScript之间的关…...

Python | Leetcode Python题解之第539题最小时间差
题目: 题解: def getMinutes(t: str) -> int:return ((ord(t[0]) - ord(0)) * 10 ord(t[1]) - ord(0)) * 60 (ord(t[3]) - ord(0)) * 10 ord(t[4]) - ord(0)class Solution:def findMinDifference(self, timePoints: List[str]) -> int:n len…...

Zookeeper运维秘籍:四字命令基础、详解及业务应用全解析
文章目录 一、四字命令基础二、四字命令详解三、四字命令的开启与配置四、结合业务解读四字命令confconsenvi命令Stat命令MNTR命令ruok命令dump命令wchswchp ZooKeeper,作为一款分布式协调服务,提供了丰富的四字命令(也称为四字短语ÿ…...

Error: `slot-scope` are deprecated报错解决
本人新手菜鸡,文章为自己遇到问题的记录,如有错误或不足还请大佬批评指正 问题描述 在Vue3环境下使用slot插槽,出现‘slot-scope’ are deprecated报错问题,经过查找发现,是因为在slot插槽使用中,vue2和vu…...

Excel(图例)中使用上标下标
单元格中 1、在Excel单元格中刷黑要设置成上标的字符,如m2中的2; 2、单击右键,在弹出的对话框中选择“设置单元格格式”; 3、在弹出的“设置单元格格式”对话框中选择上标(或下标); 4、最后…...

熔断和降级
目录 隔离和降级 FeignClient整合Sentinel 通过Feign设置服务降级 1.创建类实现FallbackFactory接口,并让这个类和使用FeignClient的接口类绑定 2.让order-service服务的feign开启sentinel 3.测试,只开启order-service服务,而不开启user-…...

【学习笔记】Linux系统基础知识 6 —— su命令详解
提示:学习Linux系统基础命令 su 命令详解,包含通过 su 命令切换用户实例 一、前期准备 1.已经正确安装并成功进入Linux系统 说明:本实验采用的 Redhat 系统(因系统不一致,可能部分显示存在差异) 二、学…...

docker-compose命令介绍
docker-compose命令介绍 docker-compose1. docker-compose是什么2. Compose file format3. 命令3.1 服务相关命令upruncreatestartrestartdownstopkillrmpauseunpause 3.2 镜像相关命令3.3 查看相关命令 docker-compose 学了docker,然后就直接去学k8s了。恍恍惚惚几…...

Spring学习笔记_29——@Transactional
Transactional 1. 介绍 Transactional 是 Spring 框架提供的一个注解,用于声明方法或类级别的事务属性。 Spring事务:Spring学习笔记_28——事务-CSDN博客 当你在一个方法或类上使用 Transactional 注解时,Spring 会为该方法或类创建一个…...

github使用基础
要通过终端绑定GitHub账号并进行文件传输,你需要使用Git和SSH密钥来实现安全连接和操作。以下是一个基本流程: 设置GitHub和SSH 检查Git安装 通过终端输入以下命令查看是否安装Git: bash 复制代码 git --version配置Git用户名和邮箱 bash …...

Flink-Kafka-Connector
Apache Flink 是一个用于处理无界和有界数据的开源流处理框架。它支持高吞吐量、低延迟以及精确一次的状态一致性等特性。Flink 社区提供了丰富的连接器(Connectors)以方便与不同的数据源进行交互,其中就包括了 Apache Kafka 连接器。 Apach…...

远程终端vim里使用系统剪切板
1、本地通过终端远程linux server,由于不是桌面环境/GUI,终端vim里似乎没办法直接使用系统剪切板,即便已经是clipboard。 $ vim --version | grep clipboard clipboard keymap printer vertsplit eval …...

底层视角看C语言
文章目录 main函数很普通main函数之前调用了什么main函数和自定义函数的对比 变量名只为人而存在goto是循环的本质指针变量指针是一个特殊的数字汇编层面看指针 数组和指针数组越界问题低端地址越界高端地址越界 引用就是指针 main函数很普通 main函数是第一个被调用的函数吗&…...

【点云学习笔记】——分割任务学习
3D点云实例分割 vs 3D点云语义分割 1. 功能对比 代码1(实例分割):用于3D点云中的实例分割任务,其目标是将点云中的物体分割成独立的实例。每个实例可能属于相同类别但需要被分开,比如在自动驾驶中的多个行人、汽车&am…...

Qt——窗口
一.窗口概述 Qt 窗口是通过 QMainWindow 类来实现的。 QMainWindow是一个为用户提供主窗口程序的类,继承QWidget类,并且提供一个预定义的布局。包含一个菜单栏(menu bar),多个工具栏(tool bars࿰…...

InfluxDB性能优化指南
1. 引言 1.1 InfluxDB的简介与发展背景 InfluxDB是一个开源的时间序列数据库(TSDB),由InfluxData公司开发,专门用于处理高频率的数据写入和查询。其设计初衷是为物联网、应用程序监控、DevOps和实时分析等场景提供一个高效的存储…...