如何快速更新自己的技术积累?
- 在现有的项目里,深挖技术,比如用到netty可以把相关底层代码和要点都看起来。
- 如果不知道目前的努力方向,就看自己的领导或公司里技术强的人在学什么。
- 知道努力方向后不知道该怎么学,就到处去找相关资料然后练习。
- 学习以后不知道有没有学成,则可以通过面试去检验。
我个人觉得面试也像是一场全新的征程,失败和胜利都是平常之事。所以,劝各位不要因为面试失败而灰心、丧失斗志。也不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!
以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目
其中异常类 Exception 又分为运行时异常( RuntimeException )和非运行时异常。Java异常又可以 分为不受检查异常( Unchecked Exception )和检查异常( Checked Exception )。
异常之间的区别与联系
==========
1.Error
error类对象由 Java 虚拟机生成并抛出,大多数错误与代码编写者所执行的操作无关。
比如所:
Java虚拟机运行错误( Virtual MachineError ),当JVM不再有继续执行操作所需的内存资源时, 将出现 OutOfMemoryError 。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止;
还有发生在虚拟机试图执行应用时,如类定义错误( NoClassDefFoundError )、链接错误 ( LinkageError )。这些错误是不可查的,因为它们在应用程序的控制和处理能力之 外,而且绝大 多数是程序运行时不允许出现的状况。
对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。在Java中,错误通常是使用 Error 的子类描述。
2.Exception
在Exception 分支中有一个重要的子类 RuntimeException (运行时异常),该类型的异常自动 为你所编写的程序定义 ArrayIndexOutOfBoundsException (数组下标越界)NullPointerException (空指针异常)、ArithmeticException (算术异常)、 MissingResourceException (丢失资源)、 ClassNotFoundException (找不到类)等异常,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。
这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生;而 RuntimeException 之外的异常我们统称为非运行时异常,类型上属于 Exception 类及其子类, 从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException 、 SQLException 等以及用户自定义的 Exception 异常,一般情况下不自定义检查异常。
注意:Error 和 Exception 的区别: Error 通常是灾难性的致命的错误,是程序无法控制和处理的,当出现这些异常时,Java虚拟机(JVM)一般会选择终止线程; Exception 通常情况下是可 以被程序处理的,并且在程序中应该尽可能的去处理这些异常。
3.检查异常和不受检查异常
检查异常:在正确的程序运行过程中,很容易出现的、情理可容的异常状况,在一定程度上这种异常的发生是可以预测的,并且一旦发生该种异常,就必须采取某种方式进行处理。
解析:除了RuntimeException及其子类以外,其他的Exception类及其子类都属于检查异常,当程序中可能出现这类异常,要么使用try-catch语句进行捕获,要么用throws子句抛出,否则编译无法通过。
不接受检查异常:包括RuntimeException及其子类和****Error。
分析:不受检查异常 为编译器不要求强制处理的异常, 检查异常 则是编译器要求必须处置的异常。
Java处理异常
========
1.try-catch
try{
//code that might generate exceptions
}catch(Exception e){
//the code of handling exception1
}catch(Exception e){
//the code of handling exception2
}
要明白异常捕获,还要理解 监控区域 (guarded region)的概念。它是一段可能产生异常的代码,并且后面跟着处理这些异常的代码。
因而可知,上述 try-catch 所描述的即是监控区域,关键词 try 后的一对大括号将一块可能发生异常的代码包起来,即为监控区域。Java方法在运行过程中发生了异常,则创建异常对象。
将异常抛出监控区域之外,由Java运行时系统负责寻找匹配的 catch 子句来捕获异常。若有一个 catch 语句匹配到了,则执行该 catch 块中的异常处理代码,就不再尝试匹配别的 catch 块了。
匹配原则:如果抛出的异常对象属于 catch 子句的异常类,或者属于该异常类的子类,则认为生成 的异常对象与 catch 块捕获的异常类型相匹配。
public class TestException {
public static void main(String[] args) {
int a = 1; int b = 0;
try {
// try监控区域
if (b == 0) throw new ArithmeticException();
// 通过throw语句抛出 异常
System.out.println(“a/b的值是:” + a / b);
System.out.println(“this will not be printed!”);
}catch (ArithmeticException e) {
// catch捕捉异常
System.out.println(“程序出现异常,变量b不能为0!”);
}
System.out.println(“程序正常结束。”);
}
}
//输出
程序出现异常,变量b不能为0!
程序正常结束。
注意:显示一个异常的描述, Throwable 重载了 toString() 方法(由 Object 定义),所以它将返回一个包含异常描述的字符串。例如,将前面的 catch 块重写成:
catch (ArithmeticException e) {
// catch捕捉异常
System.out.println(“程序出现异常”+e);
}
//输出 程序出现异常java.lang.ArithmeticException
程序正常结束。
算数异常属于运行时异常,因而实际上该异常不需要程序抛出,运行时系统自动抛出。如果不用try-catch程序就不会往下执行了。
public class TestException {
public static void main(String[] args) {
int a = 1; int b = 0;
System.out.println(“a/b的值是:” + a / b);
System.out.println(“this will not be printed!”);
}
}
结果: Exception in thread “main” java.lang.ArithmeticException: / by zero
at TestException.main(TestException.java:7)
使用多重的catch****语句:很多情况下,由单个的代码段可能引起多个异常。处理这种情况,我们需要定义两个或者更多的 catch 子句,每个子句捕获一种类型的异常,当异常被引发时,每个 catch 子句被依次检查,第一个匹配异常类型的子句执行,当一个 catch 子句执行以后,其他的子句将被旁路。
编写多重catch语句块注意事项:
**顺序问题:**先小后大,即先子类后父类
注意:
Java通过异常类描述异常类型。对于有多个 catch 子句的异常程序而言,应该尽量将捕获底层异常类的 catch 子句放在前面,同时尽量将捕获相对高层的异常类的 catch 子句放在后面。否则,捕获 底层异常类的 catch 子句将可能会被屏蔽。
**嵌套 try 语句:**try 语句可以被嵌套。也就是说,一个 try 语句可以在另一个 try 块的内部。每次进入 try 语句,异常的前后关系都会被推入堆栈。如果一个内部的 try 语句不含特殊异常的catch 处理程序,堆栈将弹出,下一个 try 语句的 catch 处理程序将检查是否与之匹配。这个过程将继续直到一个 catch 语句被匹配成功,或者是直到所有的嵌套 try 语句被检查完毕。如果没有 catch 语句匹配,Java运行时系统将处理这个异常。
class NestTry{
public static void main(String[] args){
try{
int a = args.length;
int b = 42 / a;
System.out.println("a = "+ a);
try{if(a == 1){
a = a/(a-a);
}
if(a == 2){
int c[] = {1};
c[42] =99;
}
}catch(ArrayIndexOutOfBoundsException e){
System.out.println(“ArrayIndexOutOfBounds :”+e);
}
}catch(ArithmeticException e){
System.out.println(“Divide by 0”+ e);
}
}
}
//分析运行:
D:\java>java NestTry one
a = 1
Divide by 0java.lang.ArithmeticException: / by zero D:\java>java NestTry one two
a = 2
ArrayIndexOutOfBounds :java.lang.ArrayIndexOutOfBoundsException: 42
分析:正如程序中所显示的,该程序在一个try块中嵌套了另一个 try 块。程序工作如下:当你在没 有命令行参数的情况下执行该程序,外面的 try 块将产生一个被0除的异常。
程序在有一个命令行参数条件下执行,由嵌套的 try 块产生一个被0除的异常,由于内部的 catch 块不匹配这个异常,它将把异常传给外部的 try 块,在外部异常被处理。如果你在具有两个命令行参数的条件下执行该程序,将由内部 try 块产生一个数组边界异常。
注意:当有方法调用时, try 语句的嵌套可以很隐蔽的发生。例如,我们可以将对方法的调用放在一个 try 块中。在该方法的内部,有另一个 try 语句。
在这种情况下,方法内部的 try 仍然是嵌套在外部调用该方法的 try 块中的。下面我们将对上述例子进行修改,嵌套的 try 块移到方法nesttry()的内部:结果依旧相同!
class NestTry{
static void nesttry(int a){
try{if(a == 1){
a = a/(a-a);
}
if(a == 2){
int c[] = {1};
c[42] =99;
}
}catch(ArrayIndexOutOfBoundsException e){
System.out.println(“ArrayIndexOutOfBounds :”+e);
}
}
public static void main(String[] args){
try{
int a = args.length;
int b = 42 / a;
System.out.println("a = "+ a);
nesttry(a); }catch(ArithmeticException e){
System.out.println(“Divide by 0”+ e);
}
}
}
2.throw
到目前为止,我们只是获取了被Java运行时系统引发的异常。然而,我们还可以用 throw 语句抛出明确的异常。
语法形式:
throw ThrowableInstance;
这里的ThrowableInstance一定是 Throwable 类类型或者 Throwable 子类类型的一个对象。简单 的数据类型,例如 int , char ,以及非 Throwable 类,例如 String 或 Object ,不能用作异常。
有两种方法可以获取 Throwable 对象:在 catch 子句中使用参数或者使用 new 操作符创建。程序执行完 throw 语句之后立即停止; throw 后面的任何语句不被执行,最邻近的 try 块用来检查它是否含有一个与异常类型匹配的 catch 语句。
如果发现了匹配的块,控制转向该语句;如果没有发现,次包围的 try 块来检查,以此类推。如果没有发现匹配的 catch 块,默认异常处理程序中断程序的执行并且打印堆栈轨迹。
class TestThrow{
static void proc(){
try{
throw new NullPointerException(“demo”);
}catch(NullPointerException e){
System.out.println(“Caught inside proc”);
throw e;
}
}
public static void main(String [] args){
try{proc();
}catch(NullPointerException e){
System.out.println("Recaught: "+e);
}
}
}
该程序两次处理相同的错误,首先, main() 方法设立了一个异常关系然后调用proc()。proc()方法设立了另一个异常处理关系并且立即抛出一个 NullPointerException 实例,NullPointerException 在 main() 中被再次捕获。
该程序阐述了怎样创建Java的标准异常对象,特别注意这一行:
throw new NullPointerException(“demo”);
分析:此处 new 用来构造一个 NullPointerException 实例,所有的Java内置的运行时异常有两
个构造方法:一个没有参数,一个带有一个字符串参数。
当用第二种形式时,参数指定描述异常的字符串。如果对象用作 print() 或者 println() 的参数时,该字符串被显示。这同样可以通过调用getMessage()来实现,getMessage()是由 Throwable 定义的。
3.throws
如果一个方法可以导致一个异常但不处理它,它必须指定这种行为以使方法的调用者可以保护它们自己而不发生异常。要做到这点,我们可以在方法声明中包含一个 throws 子句。
一个throws 子句列举了一个方法可能引发的所有异常类型。这对于除了 Error 或RuntimeException 及它们子类以外类型的所有异常是必要的。一个方法可以引发的所有其他类型的异常必须在 throws 子句中声明,否则会导致编译错误。
public void info() throws Exception {
//body of method
}
Exception是该方法可能引发的所有的异常,也可以是异常列表,中间以逗号隔开。
【例子】
class TestThrows{
static void throw1(){
System.out.println("Inside throw1 . ");
throw new IllegalAccessException(“demo”);
}
public static void main(String[] args){ throw1();
}
}
该例子中存在两个错误,首先,throw1()方法不想处理所导致的异常,因而它必须声明 throws 子句来列举可能引发的异常即 IllegalAccessException ;其次, main() 方法必须定义try/catch 语句来捕获该异常。
正确例子如下:
class TestThrows{
static void throw1() throws IllegalAccessException {
System.out.println("Inside throw1 . ");
throw new IllegalAccessException(“demo”);
}
总结
我个人认为,如果你想靠着背面试题来获得心仪的offer,用癞蛤蟆想吃天鹅肉形容完全不过分。想必大家能感受到面试越来越难,想找到心仪的工作也是越来越难,高薪工作羡慕不来,却又对自己目前的薪资不太满意,工作几年甚至连一个应届生的薪资都比不上,终究是错付了,错付了自己没有去提升技术。
这些面试题分享给大家的目的,其实是希望大家通过大厂面试题分析自己的技术栈,给自己梳理一个更加明确的学习方向,当你准备好去面试大厂,你心里有底,大概知道面试官会问多广,多深,避免面试的时候一问三不知。
大家可以把Java基础,JVM,并发编程,MySQL,Redis,Spring,Spring cloud等等做一个知识总结以及延伸,再去进行操作,不然光记是学不会的,这里我也提供一些脑图分享给大家:
希望你看完这篇文章后,不要犹豫,抓紧学习,复习知识,准备在明年的金三银四拿到心仪的offer,加油,打工人!
家的目的,其实是希望大家通过大厂面试题分析自己的技术栈,给自己梳理一个更加明确的学习方向,当你准备好去面试大厂,你心里有底,大概知道面试官会问多广,多深,避免面试的时候一问三不知。
大家可以把Java基础,JVM,并发编程,MySQL,Redis,Spring,Spring cloud等等做一个知识总结以及延伸,再去进行操作,不然光记是学不会的,这里我也提供一些脑图分享给大家:
[外链图片转存中…(img-jYtpSWcv-1715631220753)]
[外链图片转存中…(img-EsGGQcR7-1715631220754)]
[外链图片转存中…(img-1036y9Pl-1715631220754)]
希望你看完这篇文章后,不要犹豫,抓紧学习,复习知识,准备在明年的金三银四拿到心仪的offer,加油,打工人!