java常见的异常及例子构造

Java的常见异常及Demo

整理了一下Java常见的异常类型,并尽量在每一个附上例子,帮助清晰理解。

 

  • ArithmeticException

算术异常类,常见出现在公式计算除0中,举例来说,

 

  • ArrayIndexOutOfBoundsException 

数组下标越界异常,通常出现在数组取值时,取数据下标在数组中并不存在导致。

 

  • NullPointerException 空指针异常

最常见的异常类型,通常为指针指向null导致,可能出现在数组,循环,链表等一系列遍历,取值过程中。当对象为null时的任何操作,都会导致空指针异常。

 

  • StringIndexOutOfBoundsException

字符串越界异常

 

  • NegativeArraySizeException

建立元素个数为负数的数组异常类

 

  • ArrayStoreException

数组类型不兼容异常,通常在不兼容值赋给数组元素时抛出,如将Integer赋值给字符串:

 

  • ClassCastException 强制类型转换异常

 

  • NumberFormatException

强制类型转换错误:字符串转换为数字异常

 

ClassNotFoundException和NoClassDefFoundError的区别

 

ClassNotFoundException是一个运行时异常。从类继承层次上来看,ClassNotFoundException是从Exception继承的,所以ClassNotFoundException是一个检查异常。

当应用程序运行的过程中尝试使用类加载器去加载Class文件的时候,如果没有在classpath中查找到指定的类,就会抛出ClassNotFoundException。一般情况下,当我们使用Class.forName()或者ClassLoader.loadClass以及使用ClassLoader.findSystemClass()在运行时加载类的时候,如果类没有被找到,那么就会导致JVM抛出ClassNotFoundException。

  最简单的,当我们使用JDBC去连接数据库的时候,我们一般会使用Class.forName()的方式去加载JDBC的驱动,如果我们没有将驱动放到应用的classpath下,那么会导致运行时找不到类,所以运行Class.forName()会抛出ClassNotFoundException。

 

  • EOFException 文件已结束异常 继承自IOException

 

通过这个API,我们可以得出以下信息:

  • 这是一个IO异常的子类,名字也是END OF FILE的缩写,当然也表示流的末尾
  • 它在表明一个信息,流已经到末尾了,而大部分做法是以特殊值的形式返回给我们,而不是抛异常

  也就是说这个异常是被主动抛出来的,而不是底层或者编译器返回给我的,就像NullPointerException或IndexOutOfBoundsException一样。

  我们先来看InputStream,这个输入流,当读到了结尾会怎么样,看看API介绍:

  可以看到如果到达流的末尾,那么会返回-1,也就是说我们可以根据这个-1来判断是否到达流的末尾。

  同样的我们看一下输入流的包装类BufferedReader,它有一个读一行的方法:

  

也可以发现当读到流的末尾,通过返回值null来告诉我们到达流的末尾了,也就是说通过返回一个不可能的值来表示到达流的末尾。

 可以感觉到EOFException的用意,因为我们可以往流中放入null值,所以我们没法找到一个不可能的值来表示到达流的末尾,所以只能通过抛异常的方式来告诉用户到达末尾了。

所以说ObjectInputStream可以自己判断流是否到达末尾,但是它无法告诉我们,我们不能替代他们读取这个标记,不然ObjectInputStream将识别不了下一个内容的实际类型。

  所以呢,对于这种异常的一般解决方法就是,捕获,可以记录日志,也可以不做处理,捕获异常以后,把之前读到的数据进行后续的处理就可以了,因为那就是所以的数据。还有就是如果打算记录日志,不要把它的堆栈信息打印出来,容易给人以错觉。毕竟EOFException实质上只是一个消息而已。

  当然抛异常的做法还是有一些偏激,但是当ObjectInputStream在不知道读取对象数量的情况下,确实无法判断是否读完,除非你把之前写入对象流的数量记录下来。所以说出现这个异常时就认真分析一下,这个异常是不是代表一个信息。

 

  • FileNotFoundException 找不到文件异常

 

 

 

  • SQLException

当使用 JDBC 与数据源进行交互的时候遇见错误的时候,将会抛出名为 SQLException 的异常。一个 SQLException 的异常里面包含以下信息,用于帮助我们更好的定位错误。

 

  • IllegalAccessException

安全权限异常,一般来说,是由于java在反射时调用了private方法所导致的。

 

构造参见另一篇文章:ava反射--Field用法实践与IllegalArgumentException构造

 

  • InstantiationException

实例化异常。当试图通过newInstance()方法创建某个类的实例,而该类是一个抽象类或接口时,抛出该异常。

 

  •  NoSuchFieldException

getField(String name) 或getFields() 获取非 public 的变量,编译器会报 java.lang.NoSuchFieldException 。

 

set(Object obj, Object value) 时,变量访问检查导致的 IllegalAccessException。由于 Field 继承自 AccessibleObject , 我们可以使用 AccessibleObject.setAccessible() 方法告诉安全机制,这个变量可以访问即可解决,如field.setAccessible(true)。

 

  • java.lang.StackOverflowError和java.lang.OutOfMemoryError错误

在java虚拟机规范中,虚拟机栈和本地方法栈都会出现StackOverflowError和OutofMemoryError,程序计数器是java虚拟机中唯一一块不会产生error的内存区域。

StackOverflowError代表的是,当栈深度超过虚拟机分配给线程的栈大小时就会出现此error。

OutofMemoryError代表的是,当再申请新的内存时,虚拟机分配给线程的内存大小中无法再分配新的内存,就会出现此error。

Java内存管理详细说明移步:java内存区域与内存溢出异常(内存泄漏)

 

在单线程操作中,无论是栈深度无限增加,还是栈帧(每个方法调用执行时都会在栈中创建一个栈帧,用来存储局部变量,操作数栈,动态链表,方法出口等信息)占的空间太大,都出现的是StackOverflowError。

 

不断创建新的线程会出现OutofMemoryError的错误,这个就不要run了,会把系统内存打满。

  • 7
    点赞
  • 61
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Java中,拷贝构造函数并不像C++中那样直接支持。但我们可以通过实现一个类的 clone() 方法来实现拷贝构造函数的功能。clone() 方法是 Java 中提供的一种浅拷贝(Shallow Copy)的方法,它可以用于创建一个对象的拷贝。 下面是一个示例代码,演示了如何实现一个拷贝构造函数: ``` public class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } // 拷贝构造函数 public Person(Person another) { this.name = another.name; this.age = another.age; } // clone() 方法 public Person clone() { try { return (Person) super.clone(); } catch (CloneNotSupportedException e) { // 如果类没有实现 Cloneable 接口,则会抛出该异常 return null; } } } ``` 在上面的代码中,我们定义了一个 Person 类,并实现了一个拷贝构造函数。拷贝构造函数接受一个 Person 类型的参数,用于创建一个新的 Person 对象,该对象与传入的参数对象具有相同的属性值。 此外,我们还实现了 clone() 方法,该方法可以用于创建一个 Person 对象的拷贝。在该方法中,我们调用了 Object 类中的 clone() 方法,并将其返回类型转换为 Person 类型。 需要注意的是,clone() 方法是 Object 类中的方法,如果需要使用该方法,必须在 Person 类中实现 Cloneable 接口。如果没有实现该接口,则在调用 clone() 方法时会抛出 CloneNotSupportedException 异常。 ### 回答2: 在Java中,拷贝构造函数是一种特殊的构造函数,它用于创建一个对象的副本。拷贝构造函数允许我们将一个对象的值复制到另一个对象中,而不是仅仅复制其引用。这在需要克隆或复制对象时非常有用。 要创建一个拷贝构造函数,我们需要在类中定义一个与类名相同的方法,并传入一个参数为该类的对象。例如,假设我们有一个名为Person的类,其中包含name和age两个属性。 public class Person { private String name; private int age; public Person(Person other) { this.name = other.name; this.age = other.age; } } 在上面的例子中,我们创建了一个在类中定义的拷贝构造函数。在该函数中,我们使用另一个Person对象的属性值来初始化新对象的属性。这样一来,我们就可以通过传递一个现有的Person对象来创建一个新的Person对象。 使用拷贝构造函数有几个好处。首先,它允许我们创建一个新对象,该对象与现有对象具有相同的属性值。这对于不修改现有对象但需要创建新对象的情况非常有用。其次,它避免了浅拷贝(只复制引用而不复制实际值)可能导致的问题。 在Java中,拷贝构造函数的使用是可选的。如果我们不定义一个拷贝构造函数,Java默认提供一个浅拷贝的拷贝构造函数。但是,如果我们需要执行深拷贝(复制对象及其所有子对象的值),就需要自己定义拷贝构造函数。 总之,拷贝构造函数是在Java中创建一个对象副本的一种特殊函数。它允许我们通过将一个对象的属性值复制到另一个对象中来创建新对象。这在复制、克隆对象或创建新对象时非常有用。 ### 回答3: 在Java中,拷贝构造函数是一种特殊的构造函数,用于创建一个新对象,该对象的内容与现有对象相同。拷贝构造函数的主要作用是复制一个对象的值到另一个新的对象中。 在Java中,拷贝构造函数的定义如下: public class MyClass { private int myInt; // 拷贝构造函数 public MyClass(MyClass obj) { this.myInt = obj.myInt; } // 构造函数 public MyClass(int myInt) { this.myInt = myInt; } // Getter和Setter方法 public int getMyInt() { return myInt; } public void setMyInt(int myInt) { this.myInt = myInt; } } 在上面的示例中,我们定义了一个名为MyClass的类,该类具有一个整型成员变量myInt。我们使用两个构造函数来创建对象,一个是拷贝构造函数,一个是普通构造函数。 拷贝构造函数的参数类型是与该类相同的对象,通过该参数将现有对象的值复制到新对象中。在拷贝构造函数中,我们可以使用this关键字引用当前对象,使用点操作符(.)访问对象的成员变量。 使用拷贝构造函数创建对象的示例代码如下: MyClass obj1 = new MyClass(10); // 使用普通构造函数创建对象 MyClass obj2 = new MyClass(obj1); // 使用拷贝构造函数创建对象 System.out.println(obj1.getMyInt()); // 输出:10 System.out.println(obj2.getMyInt()); // 输出:10 通过以上代码,我们可以看到obj2对象中的myInt值与obj1对象中的myInt值相同,这是因为拷贝构造函数将obj1对象的值复制给了obj2对象。 总之,拷贝构造函数是一种创建一个新对象并复制现有对象值的常用技术,它在Java中的定义和使用方法如上所述。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值