Java大学问——优雅地处理异常,字节跳动Java研发岗这些知识点内部泄露出来了

while ((b = is.read()) != -1) {}

} catch (IOException e) {

e.printStackTrace();

} finally {

try {

is.close();

} catch (IOException e) {

e.printStackTrace();

}

}

}

总之,我们需要另外一种更优雅的解决方案。JDK7 新增了Try-With-Resource语法:如果一个类(比如InputStream)实现了 AutoCloseable接口,那么就可以将该类的对象创建在 try 关键字后面的括号中,当try-catch代码块执行完毕后,Java 会确保该对象的 close方法被调用。示例如下。

public static void main(String[] args) {

try (InputStream is = new FileInputStream(“Java高级架构狮.txt”)) {

int b;

while ((b = is.read()) != -1) {

}

} catch (IOException e) {

e.printStackTrace();

}

}

四.建议

====

关于异常处理机制的使用,我这里总结了一些非常实用的建议,希望你能够采纳。

1.尽量捕获原始的异常。


实际应该捕获 FileNotFoundException,却捕获了泛化的 Exception。示例如下。

InputStream is = null;

try {

is = new FileInputStream(“Java高级架构狮.txt”);

}

catch (Exception e) {

e.printStackTrace();

}

这样做的坏处显而易见:假如你喊“王二”,那么我就敢答应;假如你喊“老王”,那么我还真不敢答应,万一你喊的我妹妹“王三”呢?

很多初学者误以为捕获泛化的Exception更省事,但也更容易让人“丈二和尚摸不着头脑”。相反,捕获原始的异常能够让协作者更轻松地辨识异常类型,更容易找出问题的根源。

2.尽量不要打印堆栈后再抛出异常


当异常发生时打印它,然后重新抛出它,以便调用者能够适当地处理它。就像下面这段代码一样。

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

try (InputStream is = new FileInputStream(“Java高级架构狮.txt”)) {

}catch (IOException e) {

e.printStackTrace();

throw e;

}

}

这似乎考虑得很周全,但是这样做的坏处是调用者可能也打印了异常,重复的打印信息会增添排查问题的难度。

java.io.FileNotFoundException: Java高级架构狮.txt (系统找不到指定的文件。)

at java.io.FileInputStream.open0(Native Method)

at java.io.FileInputStream.open(FileInputStream.java:195)

at java.io.FileInputStream.(FileInputStream.java:138)

at java.io.FileInputStream.(FileInputStream.java:93)

at learning.Test.main(Test.java:10)

Exception in thread “main” java.io.FileNotFoundException: Java高级架构狮.txt (系统找不到指定的文件。)

at java.io.FileInputStream.open0(Native Method)

at java.io.FileInputStream.open(FileInputStream.java:195)

at java.io.FileInputStream.(FileInputStream.java:138)

at java.io.FileInputStream.(FileInputStream.java:93)

at learning.Test.main(Test.java:10)

3.千万不要用异常处理机制代替判断


我曾见过类似下面这样奇葩的代码,本来应该判 null 的,结果使用了异常处理机制来代替。

public static void main(String[] args) {

try {

String str = null;

String[] strs = str.split(“,”);

} catch (NullPointerException e) {

e.printStackTrace();

}

}

捕获异常相对判断花费的时间要多得多!我们可以模拟两个代码片段来对比一下。

代码片段 A:

long a = System.currentTimeMillis();

for (int i = 0; i < 100000; i++) {

try {

String str = null;

String[] strs = str.split(“,”);

} catch (NullPointerException e) {

}

}

long b = System.currentTimeMillis();

System.out.println(b - a);

代码片段 B:

long a = System.currentTimeMillis();

for (int i = 0; i < 100000; i++) {

String str = null;

if (str != null) {

String[] strs = str.split(“,”);

}

}

long b = System.currentTimeMillis();

System.out.println(b - a);

100000 万次的循环,代码片段 A(异常处理机制)执行的时间大概需要 1983 毫秒;代码片段 B(正常判断)执行的时间大概只需要 1 毫秒。这样的比较虽然不够精确,但足以说明问题。

4.不要盲目地过早捕获异常


如果盲目地过早捕获异常的话,通常会导致更严重的错误和其他异常。请看下面的例子。

InputStream is = null;

try {

is = new FileInputStream(“Java高级架构狮.txt”);

} catch (FileNotFoundException e) {

e.printStackTrace();

}

int b;

try {

while ((b = is.read()) != -1) {

}

} catch (IOException e) {

e.printStackTrace();

}

finally {

try {

is.close();

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
img

线程、数据库、算法、JVM、分布式、微服务、框架、Spring相关知识

一线互联网P7面试集锦+各种大厂面试集锦

学习笔记以及面试真题解析

源码讲义、实战项目、讲解视频,并且后续会持续更新**

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
[外链图片转存中…(img-ZCOthagE-1711650310531)]

线程、数据库、算法、JVM、分布式、微服务、框架、Spring相关知识

[外链图片转存中…(img-xnOaH9qg-1711650310531)]

一线互联网P7面试集锦+各种大厂面试集锦

[外链图片转存中…(img-l8iPS4uX-1711650310531)]

学习笔记以及面试真题解析

[外链图片转存中…(img-fVJct2t7-1711650310532)]

  • 17
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值