对于类而言,为了让客户端获取它自身的一个实例,最传统的方法就是提供一个公有
的构造器。还有一种方法,也应该在每个程序员的工具箱中占有一席之地。类可以提供一个
公有的静态工厂方法(static factory method),它只是一个返回类的实例的静态方法。
①静态工厂方法与构造器不同的第一大优势在于,它们有名称。如果构造器的参数本
身没有确切地描述正被返回的对象,那么具有适当名称的静态工厂会更容易使用,产生的
客户端代码也更易于阅读。
②静态工厂方法与构造器不同的第二大优势在于,不必在每次调用它们的时候都创
建一个新对象。
③静态工厂方法与构造器不同的第三大优势在于,它们可以返回原返回类型的任何子类
型的对象。
④静态工厂的第四大优势在于,所返回的对象的类可以随着每次调用而发生变化,这取
决于静态工厂方法的参数值。
⑤静态工厂的第五大优势在于,方法返回的对象所属的类,在编写包含该静态工厂方
法的类时可以不存在。
静态工厂方法的主要缺点在于,类如果不含公有的或者受保护的构造器,就不能被子
类化。
静态工厂方法的第二个缺点在于,程序员很难发现它们。在 API 文档中,它们没有像
构造器那样在 API 文档中明确标识出来,因此,对于提供了静态工厂方法而不是构造器的
类来说,要想查明如何实例化一个类是非常困难的。Javadoc 工具总有一天会注意到静态工
厂方法。同时,通过在类或者接口注释中关注静态工厂,并遵守标准的命名习惯,也可以弥
补这一劣势。下面是静态工厂方法的一些惯用名称。这里只列出了其中的一小部分:
口from——类型转换方法,它只有单个参数,返回该类型的一个相对应的实例,例如:
Date d = Date.from(instant);
口 of——聚合方法,带有多个参数,返回该类型的一个实例,把它们合并起来,例如:
Set<Rank> faceCards = EnumSet.of(JACK, QUEEN, KING);
口 valueof——比 from 和 of 更烦琐的一种替代方法,例如:
BigInteger prime = BigInteger.valueOf(Integer.MAX_VALUE);
口 instance 或者 getInstance——返回的实例是通过方法的(如有)参数来描述
的,但是不能说与参数具有同样的值,例如:
StackWalker luke = StackWalker.getInstance(options);
口 create 或者 newInstance——像 instance 或者 getInstance一样,但create
或者 newInstance 能够确保每次调用都返回一个新的实例,例如:
Object newArray = Array.newInstance(classObject, arrayLen);
口getlype——像getInstance一样,但是在工厂方法处于不同的类中的时候使
用。Type 表示工厂方法所返回的对象类型,例如:
FileStore fs = Files.getFileStore(path);
口newType——像 newInstance 一样,但是在工厂方法处于不同的类中的时候使用。