CategoryResourceRepost/极客时间专栏/geek/设计模式之美/设计原则与思想:规范与重构/37 | 实战二(下):重构ID生成器项目中各函数的异常处理代码.md
louzefeng bf99793fd0 del
2024-07-09 18:38:56 +00:00

16 KiB
Raw Blame History

平时进行软件设计开发的时候我们除了要保证正常情况下的逻辑运行正确之外还需要编写大量额外的代码来处理有可能出现的异常情况以保证代码在任何情况下都在我们的掌控之内不会出现非预期的运行结果。程序的bug往往都出现在一些边界条件和异常情况下所以说异常处理得好坏直接影响了代码的健壮性。全面、合理地处理各种异常能有效减少代码bug也是保证代码质量的一个重要手段。

在上一节课中我们讲解了几种异常情况的处理方式比如返回错误码、NULL值、空对象、异常对象。针对最常用的异常对象我们还重点讲解了两种异常类型的应用场景以及针对函数抛出的异常的三种处理方式直接吞掉、原封不动地抛出和包裹成新的异常抛出。

除此之外在上一节课的开头我们还针对ID生成器的代码提出了4个有关异常处理的问题。今天我们就用一节课的时间结合上一节课讲到的理论知识来逐一解答一下这几个问题。

话不多说,让我们正式开始今天的内容吧!

重构generate()函数

首先我们来看对于generate()函数,如果本机名获取失败,函数返回什么?这样的返回值是否合理?

  public String generate() {
    String substrOfHostName = getLastFieldOfHostName();
    long currentTimeMillis = System.currentTimeMillis();
    String randomString = generateRandomAlphameric(8);
    String id = String.format("%s-%d-%s",
            substrOfHostName, currentTimeMillis, randomString);
    return id;
  }

ID由三部分构成本机名、时间戳和随机数。时间戳和随机数的生成函数不会出错唯独主机名有可能获取失败。在目前的代码实现中如果主机名获取失败substrOfHostName为NULL那generate()函数会返回类似“null-16723733647-83Ab3uK6”这样的数据。如果主机名获取失败substrOfHostName为空字符串那generate()函数会返回类似“-16723733647-83Ab3uK6”这样的数据。

在异常情况下返回上面两种特殊的ID数据格式这样的做法是否合理呢这个其实很难讲我们要看具体的业务是怎么设计的。不过我更倾向于明确地将异常告知调用者。所以这里最好是抛出受检异常而非特殊值。

按照这个设计思路我们对generate()函数进行重构。重构之后的代码如下所示:

  public String generate() throws IdGenerationFailureException {
    String substrOfHostName = getLastFieldOfHostName();
    if (substrOfHostName == null || substrOfHostName.isEmpty()) {
      throw new IdGenerationFailureException("host name is empty.");
    }
    long currentTimeMillis = System.currentTimeMillis();
    String randomString = generateRandomAlphameric(8);
    String id = String.format("%s-%d-%s",
            substrOfHostName, currentTimeMillis, randomString);
    return id;
  }

重构getLastFieldOfHostName()函数

对于getLastFieldOfHostName()函数是否应该将UnknownHostException异常在函数内部吞掉try-catch并打印日志还是应该将异常继续往上抛出如果往上抛出的话是直接把UnknownHostException异常原封不动地抛出还是封装成新的异常抛出

  private String getLastFieldOfHostName() {
    String substrOfHostName = null;
    try {
      String hostName = InetAddress.getLocalHost().getHostName();
      substrOfHostName = getLastSubstrSplittedByDot(hostName);
    } catch (UnknownHostException e) {
      logger.warn("Failed to get the host name.", e);
    }
    return substrOfHostName;
 }

现在的处理方式是当主机名获取失败的时候getLastFieldOfHostName()函数返回NULL值。我们前面讲过是返回NULL值还是异常对象要看获取不到数据是正常行为还是异常行为。获取主机名失败会影响后续逻辑的处理并不是我们期望的所以它是一种异常行为。这里最好是抛出异常而非返回NULL值。

至于是直接将UnknownHostException抛出还是重新封装成新的异常抛出要看函数跟异常是否有业务相关性。getLastFieldOfHostName()函数用来获取主机名的最后一个字段UnknownHostException异常表示主机名获取失败两者算是业务相关所以可以直接将UnknownHostException抛出不需要重新包裹成新的异常。

按照上面的设计思路我们对getLastFieldOfHostName()函数进行重构。重构后的代码如下所示:

 private String getLastFieldOfHostName() throws UnknownHostException{
    String substrOfHostName = null;
    String hostName = InetAddress.getLocalHost().getHostName();
    substrOfHostName = getLastSubstrSplittedByDot(hostName);
    return substrOfHostName;
 }

getLastFieldOfHostName()函数修改之后generate()函数也要做相应的修改。我们需要在generate()函数中捕获getLastFieldOfHostName()抛出的UnknownHostException异常。当我们捕获到这个异常之后应该怎么处理呢

按照之前的分析ID生成失败的时候我们需要明确地告知调用者。所以我们不能在generate()函数中将UnknownHostException这个异常吞掉。那我们应该原封不动地抛出还是封装成新的异常抛出呢

我们选择后者。在generate()函数中我们需要捕获UnknownHostException异常并重新包裹成新的异常IdGenerationFailureException往上抛出。之所以这么做有下面三个原因。

  • 调用者在使用generate()函数的时候只需要知道它生成的是随机唯一ID并不关心ID是如何生成的。也就说是这是依赖抽象而非实现编程。如果generate()函数直接抛出UnknownHostException异常实际上是暴露了实现细节。
  • 从代码封装的角度来讲我们不希望将UnknownHostException这个比较底层的异常暴露给更上层的代码也就是调用generate()函数的代码。而且,调用者拿到这个异常的时候,并不能理解这个异常到底代表了什么,也不知道该如何处理。
  • UnknownHostException异常跟generate()函数,在业务概念上没有相关性。

按照上面的设计思路我们对generate()的函数再次进行重构。重构后的代码如下所示:

  public String generate() throws IdGenerationFailureException {
    String substrOfHostName = null;
    try {
      substrOfHostName = getLastFieldOfHostName();
    } catch (UnknownHostException e) {
      throw new IdGenerationFailureException("host name is empty.");
    }
    long currentTimeMillis = System.currentTimeMillis();
    String randomString = generateRandomAlphameric(8);
    String id = String.format("%s-%d-%s",
            substrOfHostName, currentTimeMillis, randomString);
    return id;
  }

重构getLastSubstrSplittedByDot()函数

对于getLastSubstrSplittedByDot(String hostName)函数如果hostName为NULL或者空字符串这个函数应该返回什么

  @VisibleForTesting
  protected String getLastSubstrSplittedByDot(String hostName) {
    String[] tokens = hostName.split("\\.");
    String substrOfHostName = tokens[tokens.length - 1];
    return substrOfHostName;
  }

理论上讲参数传递的正确性应该有程序员来保证我们无需做NULL值或者空字符串的判断和特殊处理。调用者本不应该把NULL值或者空字符串传递给getLastSubstrSplittedByDot()函数。如果传递了那就是code bug需要修复。但是话说回来谁也保证不了程序员就一定不会传递NULL值或者空字符串。那我们到底该不该做NULL值或空字符串的判断呢

如果函数是private类私有的只在类内部被调用完全在你自己的掌控之下自己保证在调用这个private函数的时候不要传递NULL值或空字符串就可以了。所以我们可以不在private函数中做NULL值或空字符串的判断。如果函数是public的你无法掌控会被谁调用以及如何调用有可能某个同事一时疏忽传递进了NULL值这种情况也是存在的为了尽可能提高代码的健壮性我们最好是在public函数中做NULL值或空字符串的判断。

那你可能会说getLastSubstrSplittedByDot()是protected的既不是private函数也不是public函数那要不要做NULL值或空字符串的判断呢

之所以将它设置为protected是为了方便写单元测试。不过单元测试可能要测试一些corner case比如输入是NULL值或者空字符串的情况。所以这里我们最好也加上NULL值或空字符串的判断逻辑。虽然加上有些冗余但多加些检验总归不会错的。

按照这个设计思路我们对getLastSubstrSplittedByDot()函数进行重构。重构之后的代码如下所示:

  @VisibleForTesting
  protected String getLastSubstrSplittedByDot(String hostName) {
    if (hostName == null || hostName.isEmpty()) {
      throw IllegalArgumentException("..."); //运行时异常
    }
    String[] tokens = hostName.split("\\.");
    String substrOfHostName = tokens[tokens.length - 1];
    return substrOfHostName;
  }

按照上面讲的我们在使用这个函数的时候自己也要保证不传递NULL值或者空字符串进去。所以getLastFieldOfHostName()函数的代码也要作相应的修改。修改之后的代码如下所示:

 private String getLastFieldOfHostName() throws UnknownHostException{
    String substrOfHostName = null;
    String hostName = InetAddress.getLocalHost().getHostName();
    if (hostName == null || hostName.isEmpty()) { // 此处做判断
      throw new UnknownHostException("...");
    }
    substrOfHostName = getLastSubstrSplittedByDot(hostName);
    return substrOfHostName;
 }

重构generateRandomAlphameric()函数

对于generateRandomAlphameric(int length)函数如果length < 0或length = 0这个函数应该返回什么

  @VisibleForTesting
  protected String generateRandomAlphameric(int length) {
    char[] randomChars = new char[length];
    int count = 0;
    Random random = new Random();
    while (count &lt; length) {
      int maxAscii = 'z';
      int randomAscii = random.nextInt(maxAscii);
      boolean isDigit= randomAscii &gt;= '0' &amp;&amp; randomAscii &lt;= '9';
      boolean isUppercase= randomAscii &gt;= 'A' &amp;&amp; randomAscii &lt;= 'Z';
      boolean isLowercase= randomAscii &gt;= 'a' &amp;&amp; randomAscii &lt;= 'z';
      if (isDigit|| isUppercase || isLowercase) {
        randomChars[count] = (char) (randomAscii);
        ++count;
      }
    }
    return new String(randomChars);
  }
}

我们先来看length < 0的情况。生成一个长度为负值的随机字符串是不符合常规逻辑的是一种异常行为。所以当传入的参数length < 0的时候我们抛出IllegalArgumentException异常。

我们再来看length = 0的情况。length = 0是否是异常行为呢这就看你自己怎么定义了。我们既可以把它定义为一种异常行为抛出IllegalArgumentException异常也可以把它定义为一种正常行为让函数在入参length = 0的情况下直接返回空字符串。不管选择哪种处理方式最关键的一点是要在函数注释中明确告知length = 0的情况下会返回什么样的数据。

重构之后的RandomIdGenerator代码

对RandomIdGenerator类中各个函数异常情况处理代码的重构到此就结束了。为了方便查看我把重构之后的代码重新整理之后贴在这里了。你可以对比着看一下跟你的重构思路是否一致。

public class RandomIdGenerator implements IdGenerator {
  private static final Logger logger = LoggerFactory.getLogger(RandomIdGenerator.class);

  @Override
  public String generate() throws IdGenerationFailureException {
    String substrOfHostName = null;
    try {
      substrOfHostName = getLastFieldOfHostName();
    } catch (UnknownHostException e) {
      throw new IdGenerationFailureException(&quot;...&quot;, e);
    }
    long currentTimeMillis = System.currentTimeMillis();
    String randomString = generateRandomAlphameric(8);
    String id = String.format(&quot;%s-%d-%s&quot;,
            substrOfHostName, currentTimeMillis, randomString);
    return id;
  }

  private String getLastFieldOfHostName() throws UnknownHostException{
    String substrOfHostName = null;
    String hostName = InetAddress.getLocalHost().getHostName();
    if (hostName == null || hostName.isEmpty()) {
      throw new UnknownHostException(&quot;...&quot;);
    }
    substrOfHostName = getLastSubstrSplittedByDot(hostName);
    return substrOfHostName;
  }

  @VisibleForTesting
  protected String getLastSubstrSplittedByDot(String hostName) {
    if (hostName == null || hostName.isEmpty()) {
      throw new IllegalArgumentException(&quot;...&quot;);
    }

    String[] tokens = hostName.split(&quot;\\.&quot;);
    String substrOfHostName = tokens[tokens.length - 1];
    return substrOfHostName;
  }

  @VisibleForTesting
  protected String generateRandomAlphameric(int length) {
    if (length &lt;= 0) {
      throw new IllegalArgumentException(&quot;...&quot;);
    }

    char[] randomChars = new char[length];
    int count = 0;
    Random random = new Random();
    while (count &lt; length) {
      int maxAscii = 'z';
      int randomAscii = random.nextInt(maxAscii);
      boolean isDigit= randomAscii &gt;= '0' &amp;&amp; randomAscii &lt;= '9';
      boolean isUppercase= randomAscii &gt;= 'A' &amp;&amp; randomAscii &lt;= 'Z';
      boolean isLowercase= randomAscii &gt;= 'a' &amp;&amp; randomAscii &lt;= 'z';
      if (isDigit|| isUppercase || isLowercase) {
        randomChars[count] = (char) (randomAscii);
        ++count;
      }
    }
    return new String(randomChars);
  }
}

重点回顾

好了,今天的内容到此就讲完了。我们一块来总结回顾一下,你需要重点掌握的内容。

今天的内容比较偏实战是对上节课学到的理论知识的一个应用。从今天的实战中你学到了哪些更高层的软件设计和开发思想呢我这里抛砖引玉总结了下面3点。

  • 再简单的代码,看上去再完美的代码,只要我们下功夫去推敲,总有可以优化的空间,就看你愿不愿把事情做到极致。
  • 如果你内功不够深厚理论知识不够扎实那你就很难参透开源项目的代码到底优秀在哪里。就像如果我们没有之前的理论学习没有今天我给你一点一点重构、讲解、分析只是给你最后重构好的RandomIdGenerator的代码你真的能学到它的设计精髓吗
  • 对比第34节课最初小王的IdGenerator代码和最终的RandomIdGenerator代码它们一个是“能用”一个是“好用”天壤之别。作为一名程序员起码对代码要有追求啊不然跟咸鱼有啥区别

课堂讨论

我们花了4节课的时间对一个非常简单的、不到40行的ID生成器代码做了多次迭代重构。除了刚刚我在“重点回顾”中讲到的那几点之外从这个迭代重构的过程中你还学到哪些更有价值的东西

欢迎在留言区写下你的思考和想法,和同学一起交流和分享。如果有收获,也欢迎你把这篇文章分享给你的朋友。