你在Java中使用过异常作为控制流吗?你也许不应该这么做。下面就是原因。
Java是一种通用的编程语言,有很多解决某个问题的方法。然而,有一些最佳实践需要遵循,也有一些坏的实践仍然普遍使用。
这些常见的错误实践之一是使用异常作为控制流。这应该避免,原因有二:它降低了代码作为单位时间响应的性能,并使代码可读性降低。
让我们首先通过查看下面的示例代码来了解如何使用异常作为控制流。代码的商业案例是:
public static int findAge(String name) { try { String ageAsString = findUser(name); return ageAsString.length(); } catch (NameNotFoundException e) { return 0; } } private static String findUser(String name) { if(name==null) { throw new NameNotFoundException(); } return name; }
如果客户端为findAge方法提供非空名称,则返回名称的长度,但如果用户名为空,则findUser方法将抛出NameNotFoundException,在这种情况下,findAge方法将返回0。
一个人如何能毫无例外地重构这段代码?坦率地说,有多种方法可以做到这一点,但这里只提供其中一种。
public static int findAgeNoEx(String name) { String ageAsString = findUserNoEx(name); return ageAsString.length(); } private static String findUserNoEx(String name) { if(name==null) { return ""; } return name; }
为了找出异常对性能的影响,我编写了以下代码,这段代码调用了两个实现的10万次,可以看出异常在我的英特尔酷睿i7-3630QM CPU上花费了数千毫秒。
public class ControlFlowWithExceptionOrNot { public static class NameNotFoundException extends RuntimeException { private static final long serialVersionUID = 3L; } private static final int TRIAL = 10000000; public static void main(String[] args) throws InterruptedException { long start = System.currentTimeMillis(); for (int i = 0; i < TRIAL; i++) { findAgeNoEx(null); } System.out.println("Duration :" + (System.currentTimeMillis() - start)); long start2 = System.currentTimeMillis(); for (int i = 0; i < TRIAL; i++) { findAge(null); } System.out.println("Duration :" + (System.currentTimeMillis() - start2)); }; public static int findAge(String name) { try { String ageAsString = findUser(name); return ageAsString.length(); } catch (NameNotFoundException e) { return 0; } } private static String findUser(String name) { if (name == null) { throw new NameNotFoundException(); } return name; } public static int findAgeNoEx(String name) { String ageAsString = findUserNoEx(name); return ageAsString.length(); } private static String findUserNoEx(String name) { if (name == null) { return ""; } return name; } }
输出得出:
Duration :16 Duration :6212
如果我们从可读性的角度比较这两个findAge方法,毫无例外的一个是非常清晰的,无论findUser方法返回什么,都要取它的长度,我们确信findUser方法将返回一个字符串。但是,有一个例外是有点混乱:findUser方法的返回不清楚。它可能返回一个字符串,或者抛出一个异常,并且从方法的签名来看,它是不可见的。因此,函数式编程范式不欢迎异常。
最后,如果在出现真正的异常情况时使用异常,则会更好。如果对控制流使用异常,则可能会导致性能问题,并且代码的可读性可能会降低。
希望这个能帮助你。