对于后端程序员来说,写Java程序的时候,处理异常是必须要做的事。错误处理固然重要,但是若是被错误处理占据了大部分逻辑,那么就大错特错了。
最近对这一点略有研究,稍微提一些优雅的处理异常的技巧和思路。
我们先看如下代码例子
- public void demo01(Long userId){
- String userName = getUserName(userId);
- if (StringUtils.isNotEmpty(userName)) {
- //继续之后的逻辑
- } else {
- log.info("没有获取到用户名称");
- }
- }
- 复制代码
若是执行这段代码的时候,userName
为空,会影响接下来的逻辑。可是若是只是输出一段日志,刚写代码的时候,比较熟悉,检查起来容易,但是时间久了或者后边的同学接手代码的时候就有点烦脑了。
最好是直接抛出异常,既然已经影响到接下来的逻辑了,那就抛出异常,阻断进程。如下所示:
- public void demo02(Long userId){
- String userName = getUserName(userId);
- if (StringUtils.isNotEmpty(userName)) {
- //继续之后的逻辑
- } else {
- throw new BusinessException("没有获取到用户名称");
- }
- }
- 复制代码
若是使用可控异常,那么在多层封装的时候,需要在每个catch块中声明这个异常,若是在最底层的方法中新增了一个可控制异常,那么每个调用此方法的函数都要新增一个throw语句。
就像如下代码所示,当getManager新增一个可控异常,那么demo03相对应也要新增一个catch块。调用demo03的方法也要新增。以此类推,那么就是从最底层到最高层的一个修改链。
- public void demo03(Long userId) throws Exception {
- try {
- String userName = getManager(userId);
- //继续之后的逻辑
- } catch (BusinessException e) {
- throw e;
- } catch (Exception e) {
- throw new Exception(e);
- }
- }
-
- private String getManager(Long userId) throws Exception {
- try {
- //获取当前用户的manager
- } catch (BusinessException e) {
- throw new BusinessException("获取manager失败");
- } catch (Exception e) {
- throw new Exception(e);
- }
- return "manager";
- }
- 复制代码
当抛出异常的时候,提供更多的信息,有助于找到问题并快速解决。能从错误日志中获取到足够多的信息,来准确分析发生错误的原因,是一件多么美妙的事情。
如下例子,通过错误信息可以看到是获取谁的manager失败了,查看数据库和代码逻辑可以快速定位问题。若是有其他信息,一并传入会有更大的帮助。
- public void demo04(Long userId) throws Exception {
- try {
- String userName = getManager(userId);
- //继续之后的逻辑
- } catch (Exception e) {
- throw new BusinessException("获取当前用户的manager失败,当前用户id为:"+userId);
- }
- }
- 复制代码
这一点感同身受,在代码中处理了大量的空判断,调用前人写的代码的时候,一不小心就空指针了,自己新写方法又重复了逻辑。
在如下例子中,getManagers
直接返回了从数据库获取到的数据,若是没有数据返回,那么List
对象就为null
。这样每个调用getManagers
的方法都要判断是否为空。
- public void demo05(Long userId) throws Exception {
- List<User> userList = getManagers(userId);
- if (!CollectionUtils.isEmpty(userList)) {
- for (User user : userList) {
- //继续之后的逻辑
- }
- }
- }
-
- private List<User> getManagers(Long userId) {
- List<User> users = getDataFromDB(userId);
- return users;
- }
- 复制代码
当getDataFromDB
的返回为空的时候,我们直接返回一个空list,这样调用的方法就不用都去判断是否为空了。 像这样:
- private List
getManagers(Long userId) { - List
users = getDataFromDB(userId); - if (CollectionUtils.isEmpty(users)) {
- return Collections.emptyList();
- } else {
- return users;
- }
- }
- 复制代码
当然还有更多的更优雅的处理异常的方法,各位可以可以在评论区留言