Java基础知识(十一)

14 异常<EXception>

14.1 概念

异常也就是非正常情况,比如使用空的引用、数组下标越界、内存溢出错误等,这些都是意外的情况,背离我们程序本身的意图。

Java提供了异常对象描述这类异常情况。

Java提供了异常机制来进行处理,通过异常机制来处理程序运行期间出现的错误。通过异常机制,可以更好地提升程序的健壮性。

可以将异常通俗的理解为“陷阱”,程序运行出现异常,就是掉到陷阱了。

注意:

Java异常是描述在一块代码中发生的异常情况(也就是错误)的对象。

14.2 异常处理基本流程

第一种情况:

try{

}

catch(…){

}

大部分设计良好的catch子句,应当能够分辨出异常情况,然后继续执行,就好像错误根本没有发生一样。

异常处理过程:

(1)当执行过程中遇到异常时,系统会抛出异常(对象)

(2)catch()块捕获异常并处理。

(3)没有被捕获的异常最终都将由默认处理程序进行处理。默认处理程序会显示一个描述异常的字符串,输出异常发生点的跟踪栈,并终止程序。

提示:

l 由try保护的语句必须使用花括号括起来(即,它们必须位于一个块中)。不能为单条语句使用try。catch块也不能省略花括号。

l try块中发生异常之后直接进入catch块,执行完catch块后也不会再返回到try块。因此,try块中发生异常之后的语句都不会再执行。

第二种情况:多条catch子句

try{

}

catch(…){

}

catch(…){

}

提示:

l try及其catch语句构成了一个单元。catch子句的作用域被限制在由之前try语句指定的那些语句。

l 执行了一条catch语句之后,会忽略其他catch语句,并继续执行try/catch块后面的代码。

l 当使用多条catch语句时,要重点记住异常子类必需位于它的所有超类之前。

第三种情况:finally

try(){

}

/可以有多个catch块/

finally{

}

提示:

l 不管是否有异常抛出,都会执行finally块。

l 最多只能有一个finally块。

l 对于关闭文件句柄、以及释放资源,finally子句都是很有用的。

第四种情况:

try(){

}

catch(…){

}

catch(…){

}…

finally{

}

提示:

每个try语句至少需要有一个catch子句或一个finally子句。

14.3 嵌套的try语句

一条try语句可以位于另外一条try语句中。

如果内层的try语句没有为特定的异常提供catch处理程序,执行流程就会跳出该try语句,检查外层try语句的catch处理程序,查看异常是否匹配。这个过程会一直继续下去,直到找到了一条匹配的catch语句,或直到检查完所有的try语句。如果没有找到匹配的catch语句,则Java运行时系统会处理该异常。

提示:

当涉及方法调用时,可以能会出现不那么明显的try语句嵌套。

14.4 异常类型

14.4.1 异常继承体系

Error一般指与JVM相关的错误,如系统崩溃、虚拟机错误、动态链接失败等。

Exception有一个重要子类RuntimeException。所有RuntimeException类及其子类的实例被称为运行时(Runtime)异常。对于您编写的程序而言,这种类型的异常是自动定义的,包括除零和无效数组索引这类情况。

运行时异常之外,称为非运行时异常/编译异常。

Throwable重写了(由Object定义的)toString()方法,从而可以返回一个包含异常描述的字符串。可以使用println()语句显示这个描述。

14.4.2 异常分类

Java的异常分为两大类:checked异常和unchecked异常。

l unchecked异常(非检查异常),也称运行时异常(RuntimeException),比如常见的NullPointerException、IndexOutOfBoundsException。对于运行时异常,java编译器不要求必须进行异常捕获处理或者抛出声明,由程序员自行决定。运行异常

l checked异常(检查异常),也称非运行时异常(运行时异常以外的异常就是非运行时异常)。Java认为Checked异常都是可以被处理(修复)的异常,所以Java程序必须显式处理Checked异常,否则无法通过编译。编译异常

含义:checked异常是指程序员比较进行检查,必须进行处理。

对于Checked异常的处理方式有如下两种:

l 当前方法明确知道如何处理该异常,则应该使用try…catch块来捕获该异常,然后在对应的块中修复该异常。

l 当前方法不知道如何处理这种异常,则应该在定义该方法时声明抛出该异常。

14.4.3 常用异常

unchecked异常 运行时异常

要求:看到异常类名,要知道表示哪种错误,知道属于哪类异常

java.lang包中定义的unchecked异常

异 常含 义
ArithmeticException算术错误,例如除零
ArrayIndexOutOfBoundsException数组索引越界
ArrayStoreException使用不兼容的类型为数组元素赋值
ClassCastException无效的转换
EnumConstantNotPresentException试图使用未定义的枚举值
IllegalArgumentException使用非法实参调用方法
IllegalMonitorStateException非法的监视操作,例如等待未锁定的线程
IllegalStateException环境或应用程序处于不正确的状态
IllegalThreadStateException请求的操作与当前线程状态不兼容
IndexOutOfBoundsException某些类型的索引越界
NegativeArraySizeException使用负数长度创建数组
NullPointerException非法使用空引用
NumberFromatException字符串到数字格式的无效转换
SecurityException试图违反安全性
StringIndexOutOfBounds试图在字符串边界之外进行索引
TypeNotPresentExcepton未找到类型
UnsupportedOpetationException遇到一个不支持的操作

常用的checked异常 编译

IOExeption //输入、输出异常

FileNotFoundException //文件不存在异常

SQLException //SQL异常

java.lang包中定义的Checked异常(了解)

异 常含 义
ClassNotFoundException未找到类
CloneNotSupportedException试图复制没有实现Cloneable接口的对象
IllegalAccessException对类的访问被拒绝
InstantiationException试图为抽象类或接口创建对象
InterruptedException一个线程被另一个线程中断
NoSuchFieldException请求的字段不存在
NoSuchMethodException请求的方法不存在
ReflectiveOperationException与反射相关的异常的子类(该异常是由JDK 7新增的)

14.5 throw

也可以使用throw语句显式地抛出一个异常。

在throw语句之后的执行流会立即停止,所有后续语句都不会执行。然后检查最近的try块,查看是否存在和异常类型相匹配的catch语句。

14.6 throws

如果方法可能引发一个Checked异常,则必须在方法声明中提供throws子句列出了方法可能抛出的异常类型,从而方法的调用者能够保卫它们自己以防备该异常。

throw和throws区别:

l throw抛异常对象,应用在代码块内

l throws声明可能抛出的异常类型,在方法定义后面。

l 如果方法内使用throw抛出Checked异常对象,又没有进行try catch处理,则该方法定义同时需要使用throws指明抛出异常类型

File类

File对象既可以表示一个文件,也可以表示一个路径/目录。

15.2.1 创建File对象

File对象描述了文件/目录本身的属性。File对象用于获取和操作与磁盘文件/目录关联的信息,例如权限、时间、日期以及目录路径,并且还可以浏览子目录层次。

File(String directoryPath)

File(String directoryPath, String filename)

File(File dirObj, String filename)

File(URI uriObj)

15.2.2 File类的常用方法

l boolean exists() //File对象所表示的文件或目录是否存在

l String getName() //获取文件名或路径名

l String getPath() //将此抽象路径名转换为一个路径名字符串

l String getAbsolutePath()

l boolean isFile() //测试此抽象路径名表示的文件是否是一个标准文件。

l boolean isDirectory() //测试此抽象路径名表示的文件是否是一个目录。

l boolean createNewFile() 创建新文件,只能创建文件,如果目录不存在,则异常

l boolean mkdir() 只能创建一层目录 make dirctory

l boolean mkdirs() 可以创建多层目录

l boolean delete() 删除文件或文件夹(要求文件夹为空)

l File[ ] listFiles() 返回值类型为

l String[] list();

l boolean canWrite() 判断文件对象表示的文件/目录是否可以写入

l boolean canRead() 判断文件对象表示的文件/目录是否可以读取

l long length() 返回文件的长度

File类中重写了equals方法,比较的文件路径。

、FileInputStream、FileOutputStream(字节流)

字节输入流InputStream主要方法:

read() :从此输入流中读取一个数据字节。

read(byte[] b) :从此输入流中将最多 b.length 个字节的数据读入一个 byte 数组中。

read(byte[] b, int off, int len) :从此输入流中将最多 len 个字节的数据读入一个 byte 数组中。

close():关闭此输入流并释放与该流关联的所有系统资源。

字节输出流OutputStream主要方法:

write(byte[] b) :将 b.length 个字节从指定 byte 数组写入此文件输出流中。

write(byte[] b, int off, int len) :将指定 byte 数组中从偏移量 off 开始的 len 个字节写入此文件输出流。

write(int b) :将指定字节写入此文件输出流。

close() :关闭此输入流并释放与该流关联的所有系统资源。

字节流的方式效率较低,不建议使用

public class IOTest {

public static void main(String[] args) throws IOException {

File file = new File("D:/test.txt");

write(file);

System.out.println(read(file));

}

public static void write(File file) throws IOException {

OutputStream os = new FileOutputStream(file, true);

// 要写入的字符串

String string = "松下问童子,言师采药去。只在此山中,云深不知处。";

// 写入文件

os.write(string.getBytes());

// 关闭流

os.close();

}

public static String read(File file) throws IOException {

InputStream in = new FileInputStream(file);

// 一次性取多少个字节

byte[] bytes = new byte[1024];

// 用来接收读取的字节数组

StringBuilder sb = new StringBuilder();

// 读取到的字节数组长度,为-1时表示没有数据

int length = 0;

// 循环取数据

while ((length = in.read(bytes)) != -1) {

// 将读取的内容转换成字符串

sb.append(new String(bytes, 0, length));

}

// 关闭流

in.close();

return sb.toString();

}

}

、FileInputStream、FileOutputStream(字节流)

字节输入流InputStream主要方法:

read() :从此输入流中读取一个数据字节。

read(byte[] b) :从此输入流中将最多 b.length 个字节的数据读入一个 byte 数组中。

read(byte[] b, int off, int len) :从此输入流中将最多 len 个字节的数据读入一个 byte 数组中。

close():关闭此输入流并释放与该流关联的所有系统资源。

字节输出流OutputStream主要方法:

write(byte[] b) :将 b.length 个字节从指定 byte 数组写入此文件输出流中。

write(byte[] b, int off, int len) :将指定 byte 数组中从偏移量 off 开始的 len 个字节写入此文件输出流。

write(int b) :将指定字节写入此文件输出流。

close() :关闭此输入流并释放与该流关联的所有系统资源。

字节流的方式效率较低,不建议使用

public class IOTest {

public static void main(String[] args) throws IOException {

File file = new File("D:/test.txt");

write(file);

System.out.println(read(file));

}

public static void write(File file) throws IOException {

OutputStream os = new FileOutputStream(file, true);

// 要写入的字符串

String string = "松下问童子,言师采药去。只在此山中,云深不知处。";

// 写入文件

os.write(string.getBytes());

// 关闭流

os.close();

}

public static String read(File file) throws IOException {

InputStream in = new FileInputStream(file);

// 一次性取多少个字节

byte[] bytes = new byte[1024];

// 用来接收读取的字节数组

StringBuilder sb = new StringBuilder();

// 读取到的字节数组长度,为-1时表示没有数据

int length = 0;

// 循环取数据

while ((length = in.read(bytes)) != -1) {

// 将读取的内容转换成字符串

sb.append(new String(bytes, 0, length));

}

// 关闭流

in.close();

return sb.toString();

}

}

BufferedInputStream、BufferedOutputStream(缓冲字节流)

缓冲字节流是为高效率而设计的,真正的读写操作还是靠FileOutputStream和FileInputStream,所以其构造方法入参是这两个类的对象也就不奇怪了。

public class IOTest {

public static void write(File file) throws IOException {

// 缓冲字节流,提高了效率

BufferedOutputStream bis = new BufferedOutputStream(new FileOutputStream(file, true));

// 要写入的字符串

String string = "松下问童子,言师采药去。只在此山中,云深不知处。";

// 写入文件

bis.write(string.getBytes());

// 关闭流

bis.close();

}

public static String read(File file) throws IOException {

BufferedInputStream fis = new BufferedInputStream(new FileInputStream(file));

// 一次性取多少个字节

byte[] bytes = new byte[1024];

// 用来接收读取的字节数组

StringBuilder sb = new StringBuilder();

// 读取到的字节数组长度,为-1时表示没有数据

int length = 0;

// 循环取数据

while ((length = fis.read(bytes)) != -1) {

// 将读取的内容转换成字符串

sb.append(new String(bytes, 0, length));

}

// 关闭流

fis.close();

return sb.toString();

}

}

3、**InputStreamReader、OutputStreamWriter(字符流)**

字符流方法

字符输入流Reader主要方法:

read():读取单个字符。

read(char[] cbuf) :将字符读入数组。

read(char[] cbuf, int off, int len) : 将字符读入数组的某一部分。

read(CharBuffer target) :试图将字符读入指定的字符缓冲区。

flush() :刷新该流的缓冲。

close() :关闭此流,但要先刷新它。

字符输出流Writer主要方法:

write(char[] cbuf) :写入字符数组。

write(char[] cbuf, int off, int len) :写入字符数组的某一部分。

write(int c) :写入单个字符。

write(String str) :写入字符串。

write(String str, int off, int len) :写入字符串的某一部分。

flush() :刷新该流的缓冲。

close() :关闭此流,但要先刷新它。

另外,字符缓冲流还有两个独特的方法:

BufferedWriter类newLine() :写入一个行分隔符。这个方法会自动适配所在系统的行分隔符。

BufferedReader类readLine() :读取一个文本行。

字符流适用于文本文件的读写,OutputStreamWriter类其实也是借助FileOutputStream类实现的,故其构造方法是FileOutputStream的对象

public class IOTest {

public static void write(File file) throws IOException {

// OutputStreamWriter可以显示指定字符集,否则使用默认字符集

OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(file, true), "UTF-8");

// 要写入的字符串

String string = "松下问童子,言师采药去。只在此山中,云深不知处。";

osw.write(string);

osw.close();

}

public static String read(File file) throws IOException {

InputStreamReader isr = new InputStreamReader(new FileInputStream(file), "UTF-8");

// 字符数组:一次读取多少个字符

char[] chars = new char[1024];

// 每次读取的字符数组先append到StringBuilder中

StringBuilder sb = new StringBuilder();

// 读取到的字符数组长度,为-1时表示没有数据

int length;

// 循环取数据

while ((length = isr.read(chars)) != -1) {

// 将读取的内容转换成字符串

sb.append(chars, 0, length);

}

// 关闭流

isr.close();

return sb.toString()

}

}

//缓冲字符流

public static void write(File file) throws IOException {

// BufferedWriter fw = new BufferedWriter(new OutputStreamWriter(new

// FileOutputStream(file, true), "UTF-8"));

// FileWriter可以大幅度简化代码

BufferedWriter bw = new BufferedWriter(new FileWriter(file, true));

// 要写入的字符串

String string = "松下问童子,言师采药去。只在此山中,云深不知处。";

bw.write(string);

bw.close();

}

public static String read(File file) throws IOException {

BufferedReader br = new BufferedReader(new FileReader(file));

// 用来接收读取的字节数组

StringBuilder sb = new StringBuilder();

// 按行读数据

String line;

// 循环取数据 readLine读取一行

while ((line = br.readLine()) != null) {

// 将读取的内容转换成字符串

sb.append(line);

}

// 关闭流

br.close();

return sb.toString();

}

  • 7
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值