• 面对Java出现异常,应该怎么做才是正确的?


    对于后端程序员来说,写Java程序的时候, 处理异常 是必须要做的事。错误处理固然重要,但是若是被错误处理占据了大部分逻辑,那么就大错特错了。

    最近对这一点略有研究,稍微提一些优雅的处理异常的技巧和思路。

    1 使用异常而不是返回码

    我们先看如下代码例子

    1. <pre class="prettyprint hljs cpp" style="padding: 0.5em; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; color: rgb(68, 68, 68); border-radius: 4px; display: block; margin: 0px 0px 1.5em; font-size: 14px; line-height: 1.5em; word-break: break-all; overflow-wrap: break-word; white-space: pre; background-color: rgb(246, 246, 246); border: none; overflow-x: auto;">public void demo01(Long userId){
    2. String userName = getUserName(userId);
    3. if (StringUtils.isNotEmpty(userName)) {
    4. //继续之后的逻辑
    5. } else {
    6. log.info("没有获取到用户名称");
    7. }
    8. }

    若是执行这段代码的时候, userName 为空,会影响接下来的逻辑。可是若是只是输出一段日志,刚写代码的时候,比较熟悉,检查起来容易,但是时间久了或者后边的同学接手代码的时候就有点烦脑了。

    最好是直接抛出异常,既然已经影响到接下来的逻辑了,那就抛出异常,阻断进程。如下所示:

    1. <pre class="prettyprint hljs cpp" style="padding: 0.5em; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; color: rgb(68, 68, 68); border-radius: 4px; display: block; margin: 0px 0px 1.5em; font-size: 14px; line-height: 1.5em; word-break: break-all; overflow-wrap: break-word; white-space: pre; background-color: rgb(246, 246, 246); border: none; overflow-x: auto;">public void demo02(Long userId){
    2. String userName = getUserName(userId);
    3. if (StringUtils.isNotEmpty(userName)) {
    4. //继续之后的逻辑
    5. } else {
    6. throw new BusinessException("没有获取到用户名称");
    7. }
    8. }

    2 使用不可控异常

    若是使用可控异常,那么在多层封装的时候,需要在每个catch块中 声明 这个异常,若是在最底层的方法中新增了一个可控制异常,那么每个调用此方法的函数都要新增一个throw语句。

    就像如下代码所示,当getManager新增一个可控异常,那么demo03相对应也要新增一个catch块。调用demo03的方法也要新增。以此类推,那么就是从最底层到最高层的一个修改链。

    1. "prettyprint hljs php" style="padding: 0.5em; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; color: rgb(68, 68, 68); border-radius: 4px; display: block; margin: 0px 0px 1.5em; font-size: 14px; line-height: 1.5em; word-break: break-all; overflow-wrap: break-word; white-space: pre; background-color: rgb(246, 246, 246); border: none; overflow-x: auto;">public void demo03(Long userId) throws Exception {
    2. try {
    3. String userName = getManager(userId);
    4. //继续之后的逻辑
    5. } catch (BusinessException e) {
    6. throw e;
    7. } catch (Exception e) {
    8. throw new Exception(e);
    9. }
    10. }
    11. private String getManager(Long userId) throws Exception {
    12. try {
    13. //获取当前用户的manager
    14. } catch (BusinessException e) {
    15. throw new BusinessException("获取manager失败");
    16. } catch (Exception e) {
    17. throw new Exception(e);
    18. }
    19. return "manager";
    20. }

    3 给异常提供更多的信息

    当抛出异常的时候,提供更多的信息,有助于找到问题并快速解决。能从错误日志中获取到足够多的信息,来准确分析发生错误的原因,是一件多么美妙的事情。

    如下例子,通过错误信息可以看到是获取谁的manager失败了,查看数据库和代码逻辑可以快速定位问题。若是有其他信息,一并传入会有更大的帮助。

    1. <pre class="prettyprint hljs java" style="padding: 0.5em; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; color: rgb(68, 68, 68); border-radius: 4px; display: block; margin: 0px 0px 1.5em; font-size: 14px; line-height: 1.5em; word-break: break-all; overflow-wrap: break-word; white-space: pre; background-color: rgb(246, 246, 246); border: none; overflow-x: auto;">public void demo04(Long userId) throws Exception {
    2. try {
    3. String userName = getManager(userId);
    4. //继续之后的逻辑
    5. } catch (Exception e) {
    6. throw new BusinessException("获取当前用户的manager失败,当前用户id为:"+userId);
    7. }
    8. }

    4 别在方法中返回null值

    这一点感同身受,在代码中处理了大量的空判断,调用前人写的代码的时候,一不小心就空指针了,自己新写方法又重复了逻辑。

    在如下例子中, getManagers 直接返回了从数据库获取到的数据,若是没有数据返回,那么 List 对象就为 null 。这样每个调用 getManagers 的方法都要判断是否为空。

    1. <pre class="prettyprint hljs php" style="padding: 0.5em; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; color: rgb(68, 68, 68); border-radius: 4px; display: block; margin: 0px 0px 1.5em; font-size: 14px; line-height: 1.5em; word-break: break-all; overflow-wrap: break-word; white-space: pre; background-color: rgb(246, 246, 246); border: none; overflow-x: auto;">public void demo05(Long userId) throws Exception {
    2. List<User> userList = getManagers(userId);
    3. if (!CollectionUtils.isEmpty(userList)) {
    4. for (User user : userList) {
    5. //继续之后的逻辑
    6. }
    7. }
    8. }
    9. private List<User> getManagers(Long userId) {
    10. List<User> users = getDataFromDB(userId);
    11. return users;
    12. }

    当 getDataFromDB 的返回为空的时候,我们直接返回一个空list,这样调用的方法就不用都去判断是否为空了。 像这样:

    1. <pre class="prettyprint hljs kotlin" style="padding: 0.5em; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; color: rgb(68, 68, 68); border-radius: 4px; display: block; margin: 0px 0px 1.5em; font-size: 14px; line-height: 1.5em; word-break: break-all; overflow-wrap: break-word; white-space: pre; background-color: rgb(246, 246, 246); border: none; overflow-x: auto;">private List<User> getManagers(Long userId) {
    2. List<User> users = getDataFromDB(userId);
    3. if (CollectionUtils.isEmpty(users)) {
    4. return Collections.emptyList();
    5. } else {
    6. return users;
    7. }
    8. }
  • 相关阅读:
    [附源码]Python计算机毕业设计宠物销售管理系统
    SwiftUI4.0有趣的动画升级:新iOS16视图内容过渡动画
    Kubernetes技术--k8s核心技术 configMap
    IDEA2021.1默认设置打开方式(JDK版本和Maven版本)
    已经安装了python如何与anaconda共存?
    通过Demo学WPF—数据绑定(二)
    链表内指定区间反转
    曝光网上流传的新白薅某宝虚拟物品正规国内educn邮箱方法0风险成功率100%(附带话术
    Multi-Grade Deep Learning for Partial Differential Equations
    怒刷LeetCode的第22天(Java版)
  • 原文地址:https://blog.csdn.net/Java_ttcd/article/details/126438734