Effective Java
Effective Java
创建和销毁对象---考虑用静态工厂方法代替构造器
构造器是创建一个对象实例最基本也最通用的方法,大部分开发者在使用某个class的时候,首先需要考虑的就是如何构造和初始化一个对象示例,而构造的方式首先考虑到的就是通过构造函数来完成,因此在看javadoc中的文档时首先关注的函数也是构造器。所以对于类而言,我们为了获得一个类的实例对象,通常情况下会提供一个公有的(public) 的构造器。当然除了这种方法以外,我们还可以通过给类提供一个public的静态工厂方法(static factory method)的方式来完成,让它返回一个类的实例。如:
public static Boolean valueOf(boolean b) {
return b ? Boolean.TRUE : Boolean.FALSE;
}
这个示例将boolean基本类型值转换成一个Boolean对象的引用。(需要注意的是这里的静态工厂方法与设计模式中所说的工厂方法模式不同)那么,在Effective Java的首篇中,我们来看看静态工厂方法相比于构造器带来了哪些优势:1. 静态工厂方法可以有不同的名称,代码更清楚,更易读。在框架设计中,针对某些工具类通常会考虑空对象以辨别该对象是否已经被初始化。构造器的命名都一致,一个类只能有一个指定签名的构造器。当一个类需要提供多个构造器时,通常只是通过不同的形参类型的顺序加以区分,但其函数名还是相同的,无法提供较高的区分度。那么,当一个类需要多个带有相同签名的构造器时,不妨考虑用静态工厂方法代替构造器,并且慎重地选择名称以便突出它们之间的区别。这样该静态工厂方法有名称,通过赋予有意义的名称,使用该方法的程序员可以清晰的知道该方法的含义。(方法的签名(signature)由它的名称和所有参数的返回类型组成,而不包括它的返回类型)2. 静态工厂方法不必在每次调用它们的时候都创建一个新对象。比如你可以在静态工厂方法里限定创建该对象的个数,当超出规定的个数时,返回缓存里的对象。场景:调用构造器时每次都创建新对象。这一点很显然,我们知道Singleton其实也提供一个静态工厂方法获取实例。除此之外,可以用==代替equals()方法,达到性能的提升。3. 静态工厂方法可以返回原返回类型的任何子类型的对象。场景:构造器只能返回当前类的实例,无法返回子类的实例。这个优点的确很有用,能够充分利用多态,使代码更具有可扩展性。设计模式中的工厂方法也有体现,可根据入参type返回类型为Types的不同子类实例。
public Types getType(String type) {...}
这样我们在选择返回对象的类时就有更大的灵活性,这种灵活性的应用就是API可以返回对象,同时又不会使对象的类变成公有的。以这种方式隐藏实现类会使API变得简洁。这项技术适合于基于接口的框架(interface-based framework)。这种面向接口的编程也提高了软件的可维护性和性能。4. 静态工厂方法在创建参数化类型实例的时候使代码变得更加简洁。场景: Java的泛型类在实例化时,还是需要写两次类型参数,非常冗长。静态工厂方法可以帮你改善这种情况:举个例子,假设在HashMap类中提供如下静态工厂方法:
public static <k, v=""> HashMap<k, v=""> newInstance(){
return new HashMap<k, v="">();
}
那么,调用时应该是这样的:
HashMap<stirng, list<string="">> m = HashMap.newInstance();
而在调用参数化的构造器时,则通常需要这样做,显得比较繁琐。
HashMap<string, list<string="">> m = new HashMap<string, list<string="">>();
另外值得一提的是,静态工厂方法返回的对象所属的类,在编写包含该静态工厂方法的类时不必存在。这种灵活的静态工厂方法构成了服务提供者框架(Service Provider Framework)的基础,例如JDBC的API。所谓服务提供者框架是指这样的一个系统:多个服务提供者实现同一个服务,由框架(即系统)来负责为客户端提供这种服务的多个实现并将这些实现解耦。服务提供者框架有三个主要的组件:1.服务接口(Service interface),这是提供者实现的。2.提供者注册API(Provider Registration API),这是系统用来注册实现,让客户端访问它们的。3.服务访问API(Service Access API),这是客户端获取服务实例用的。另外还有第四个可选接口,服务提供者接口(Service Provider Interface),负责创建其服务实现的实例。若没有服务提供者接口,实现就按照类名称进行注册,并通过反射方式进行实例化。对JDBC而言,Connection就是它的服务接口,DriverManager.registerDriver就是它的提供者注册API,DriverManager.getConnection是服务访问API,Driver就是服务提供者接口。
如提供了两个静态工厂方法empty和preallocate(含义,重新分配;再指派)用于分别创建一个空对象和一个带有指定分配空间的String对象。从使用方式来看,这些静态方法确实提供了有使用者很容易就可以判断出它们的作用和应用场景,而不必在一组重载的构造器中去搜寻每一个构造函数及其参数列表,以找出适合当前场景的构造函数。
从效率方面来讲,由于提供了唯一的静态空对象,当判读对象实例是否为空时(isEmpty),直接使用预制静态空对象(emptyString)的地址与当前对象进行比较,如果是同一地址,即可确认当前实例为空对象了。对于preallocate函数,顾名思义,该函数预分配了指定大小的内存空间,后面在使用该String实例时,不必担心赋值或追加的字符过多而导致频繁的realloc(重新分配内存)等操作。2.不必在每次调用它们的时候创建一个新的对象。还是基于上面的代码实例,由于所有的空对象都共享同一个静态空对象,这样也节省了更多的内存开销,如果是strEmpty2方式构造出的空对象,在执行比较等操作时会带来更多的效率开销。事实上,Java在String对象的实现中,使用了常量资源池也是基于了同样的优化策略。该优势同样适用于单实例模式。3.可以返回原返回类型的任何子类型。在Java Collections Framework的集合接口中,提供了大量的静态方法返回集合接口类型的实现类型,如Collections.subList()、Collections.unmodifiableList()等。返回的接口是明确的,然而针对具体的实现类,函数的使用者并不也无需知晓。这样不仅极大的减少了导出类的数量,而且在今后如果发现某个子类的实现效率较低或者发现更好的数据结构和算法来替换当前实现子类时,对于集合接口的使用者来说,不会带来任何的影响。本书在例子中提到EnumSet是通过静态工厂方法返回对象实例的,没有提供任何构造函数,其内部在返回实现类时做了一个优化,即如果枚举的数量小于64,该工厂方法将返回一个经过特殊优化的实现类实例(RegularEnumSet),其内部使用long(64bits在Java中)中的不同位来表示不同的枚举值。如果枚举的数量大于64,将使用long的数组作为底层支撑。然而这些内部实现类的优化对于使用者来说是透明的。4.在创建参数化类型实例的时候,它们使代码变得更加简洁。Map<String,String> m = new HashMap<String,String>(); 由于Java在构造函数的调用中无法进行类型的推演,因此也就无法通过构造器的参数类型来实例化指定类型参数的实例化对象。然而通过静态工厂方法则可以利用参数类型推演的优势,避免了类型参数在一次声明中被多次重写所带来的烦忧,见如下代码:
Effective Java的更多相关文章
- Effective java笔记(二),所有对象的通用方法
Object类的所有非final方法(equals.hashCode.toString.clone.finalize)都要遵守通用约定(general contract),否则其它依赖于这些约定的类( ...
- 《Effective java》-----读书笔记
2015年进步很小,看的书也不是很多,感觉自己都要废了,2016是沉淀的一年,在这一年中要不断学习.看书,努力提升自己!预计在2016年要看12本书,主要涉及java基础.Spring研究.java并 ...
- 《Effective Java》学习笔记——积累和激励
从一个实际案例说起 国庆长假前一个礼拜,老大给我分配了这么一个bug,就是打印出来的报表数量为整数的,有的带小数位,有的不带,毫无规律. 根据短短的两个多月的工作经验以及猜测,最终把范围缩小到以下这段 ...
- Effective Java笔记一 创建和销毁对象
Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...
- effective java 读后感
think in java , effective java 这两本书一直都在java的生态圈中经久不衰.本来想着先翻过 think in java 这本大山,但是读到一半就放弃了.过长的篇幅,让 ...
- Effective java读书笔记
2015年进步很小,看的书也不是很多,感觉自己都要废了,2016是沉淀的一年,在这一年中要不断学习.看书,努力提升自己 计在16年要看12本书,主要涉及java基础.Spring研究.java并发.J ...
- effective java —— 终结方法守卫者
目录: effective java —— 终结方法守卫者 effective java 第2章:创建和销毁对象.第7条 : 避免使用终结方法.最后的“终结方法守卫者 (finalizer guard ...
- Effective Java 创建和销毁对象
<Effective Java>阅读笔记,用适合自己理解的方式提炼该书内容.<Effective Java>是一本很实用的书,阅读方法应该是快速的领会,总结,然后应用.而非,一 ...
- Effective Java Index
Hi guys, I am happy to tell you that I am moving to the open source world. And Java is the 1st langu ...
随机推荐
- EF架构~真正被封装的排序方法,支持多列排序
回到目录 对于linq to sql 和linq to entity来说,当你把获取数据的方法封装了之后,总觉得还缺点什么,想了之后,应该是排序,但看了微软的orchard项目之后,觉得它的排序封装的 ...
- 用css来写一些简单的图形
在写网页的过程中,有时我们需要用到一些简单的图片但是手头又没有合适的,我们其实可以自己来写,下面我就简单的介绍几个例子: 1.上三角 Triangle Up #triangle-up { width: ...
- poi 输出Excel显示内容
在业务系统中多少回接触到Excel解析.在java开发平台下选择 Apache POI是一个非常明智的选择,POI提供非常完善API来读取或写入Microsoft Office Excel. 目前对导 ...
- 构造persen
package java1; //人类 public class Person { //属性 成员变量 String name; int age=30; //方法 函数 成员函数 void hello ...
- fabric upgrade from old crashlystic stuck in build
提示build,一直没有反应. 问题:工程中原有的shell命令屏蔽掉了
- PopupWindow+ListView+OnItemClick点击无效
昨天踩了个大坑,从下午折腾到现在.实现以下功能: popupWindow显示listview,listView OnItemClick点击后获取值. 由于重写listview 是有两部分 列表正文和右 ...
- JAVA--网络编程(UDP)
上午给大家简单介绍了一下TCP网络通信的知识,现在就为大家补充完整网络编程的知识,关于UDP的通信知识. UDP是一种不可靠的网络协议,那么还有什么使用价值或必要呢?其实不然,在有些情况下UDP协议可 ...
- Java多线程系列--“JUC集合”10之 ConcurrentLinkedQueue
概要 本章对Java.util.concurrent包中的ConcurrentHashMap类进行详细的介绍.内容包括:ConcurrentLinkedQueue介绍ConcurrentLinkedQ ...
- 微信内置浏览器WebApp开发,踩坑 · Issue #31 · maxzhang/maxzhang.github.com · GitHub
最近花6天时间完成了一个七夕的小活动,是一个简单的WebApp.由于我前期对面向微信的Web开发评估不足,导致开发过程十分艰难.写这篇文章总结下,惊醒自己未来不要再犯这样的错误. 问题: 1. 有些比 ...
- Spark入门实战系列--7.Spark Streaming(下)--实时流计算Spark Streaming实战
[注]该系列文章以及使用到安装包/测试数据 可以在<倾情大奉送--Spark入门实战系列>获取 .实例演示 1.1 流数据模拟器 1.1.1 流数据说明 在实例演示中模拟实际情况,需要源源 ...