第二章 创建和销毁对象

原创 2012年03月21日 22:37:17
第一条:考虑用静态工厂方法代替构造器

 

类可以提供一个公有的静态工厂方法(static  factory method),它只是一个返回类的实例的静态方法

注意:静态工厂方法与设计模式中的工厂方法模式不同。本条目中所指的静态工厂方法并不直接对应于设计模式中的工厂方法

静态工厂方法与构造器不同的第一大优势在于,他们有名称。

如果构造器的参数本身没有确切地描述正被返回的对象,那么具有适当名称的静态工厂会更容易使用,产生的客户端代码也更易于阅读。

当一个类需要多个带有相同签名的构造器时,就用静态工厂方法代替构造器,并且慎重地选择名称以便突出它们之间的区别。

静态工厂方法与构造器不同的第二大优势在于,不必在每次调用它们的时候都创建一个新对象

这使得不可变类可以使用预先构建好的实例,或者将构建好的实例缓存起来,进行重复利用,从而避免创建不必要的重复对象。

次方法类似于Flyweight模式

如果程序经常请求创建相同的对象,并且创建对象的代价很高,则这项技术可以极大地提升性能。

静态工厂方法能够为重复的调用返回相同的对象,这样有助于类总能严格控制在某个时刻哪些实例应该存在。这种类称作实例受控的类(instance-controlled)。编写实力受控的类有几个原因。实例受控使得类可以确保它是一个Singleton或者是不可实例化的。它还使得不可变的类可以确保不会存在连个相等的实例,即当且仅当a==b的时候才有a.equals(b)为true

静态工厂方法与构造器不同的第三大优势在于,它们可以返回原返回类型的任何子类型的对象

这种灵活性的一种应用是,API可以返回对象,同时又不会使对象的类变成公有的。以这种方式隐藏实现类会使API变得非常简洁。这种技术适用于基于接口的框架(interface-based framework),因为在这种框架中,接口为静态工厂方法提供了自然返回类型

使用这种静态工厂方法时,甚至要求客户端通过接口来引用被返回的对象,而不是通过它的实现类来引用被返回的对象,这是一种良好的习惯。

公有的静态工厂方法所返回的对象的类不仅可以是非公有的,而且该类还可以随着每次调用而发生变化,这取决于静态工厂方法的参数值 。

例如:发行版本1.5中引入的类java.util.EnumSet没有公有的构造器,只有静态工厂方法。它们返回两种实现类之一,具体则取决于底层枚举类型的大小:如果它的元素有64个或者更少,就像大多数枚举类型一样,静态工厂方法就会返回一个RegalarEumSet实例,用单个long进行支持;如果枚举类型有65个或者更多元素,工厂就返回JumboEnumSet实例,用long数组进行支持。

静态工厂方法返回的对象所属的类,在编写包含该静态工厂方法的类时可以不必存在。这种灵活的静态工厂方法构成了服务提供这框架(Service Provider Framework)的基础,例如JDBC(Java Database Connectivity)API。服务提供者框架是指这样一个系统:多个服务提供者实现一个服务,系统为服务提供者的客户端提供多个实现,并把他们从多个实现中解耦出来。

服务提供者框架中有三个重要的组件:服务接口(Service Interface),这是提供者实现的;提供者注册API(Provider Registration API),这是系统用来注册实现,让用户端访问它们的;服务访问API(Service Access API),是客户端用来获取服务的实例的。服务访问API一般允许但是不要求客户端指定某种选择提供者的条件

服务提供者框架的第四个组件是可选的:服务提供者接口(Service Provider Interface),这些提供者负责创建其服务实现的实例。如果没有服务提供者接口,实现就按照类名称注册,并通过凡是方式进行实例化。对于JDBC来说,Connection就是它的服务接口,DriverManager.registerDriver是提供者注册API,DriverManager.getConnection是服务访问API,Driver就是服务提供者接口。

//Service provider framework sketh
//Service interface
public interface Service{
    ..//Service-specific methods go here
}

//Service provider interface
public interface Provider{
     Service newService();
}

//Noninstantiable class for service registration and access
public class Service{
private Services(){}//Prevents instantiation(Item 4)
//Maps service names to services
private static final Map<String,Provider> providers  = new ConcurrentHashMap<String,Provider>();
public static final String DEFAULT_PROVIDER_NAME = "<def>";
//Provider registration API
public static void registerDefaultProvider(Provider p){
    registerProvider(DEFAULT_PROVIDER_NAME,p)
}
public static void registerProvider(String name,Provider p){
   providers.put(name,p);
}

//Service accss API
public static Service newInstance(){
return newInstance(DEFAULT_PROVIDER_NAME);
}

public static Service newInstance(String name){
Provider p = providers.get(name);
if(p==null)
   throw new IllegalArgumentException("No provider registered with name:"+name);
return p.newService();}} 

静态工厂方法的第四大优势在于,在创建参数化类型实例的时候,它们使代码变得更加简洁

静态工厂方法的主要缺点在于,类如果不含公有的或者受保护的构造器,就不能被子类化。

静态工厂方法的第二个缺点在于,它们与其他的静态方法实际上没有任何区别

Effective Java 2nd 阅读 第二章 创建和销毁对象

第一条:考虑用静态工厂方法代替构造器 1:静态工厂方法可以根据不同的需要来定义方法名称,而公用的构造方法因为命名一致,所以不容易区分差异性 2:当参数一致时,如果使用公用的构造方法,则会冲突,当改变构...

effective java 读书笔记---第二章创建和销毁对象

20170405开始读 effective java 1.使用静态构造方法代替构造器 优点:能够控制类的构造,(多例,单例),有意义的构造名称,精简代码,可以返回任何子类型 缺点:类如果不含有任...

TEC1701.WebADI开发技术总结 - 第二章 Step2 - 创建数据库对象(2/6)

第二章  Step2 - 创建数据库对象 客户化开发WebADI,首先要建立好相应的数据库对象。然后再通过配置来创建需要使用的WebADI。 创建数据库对象时一定要注意Data object要生...

第二章 对象的创建和使用 笔记

关于函数声明的一点说明:形如int func();的声明,在C和C++中大有不同。在C中,表示“一个可以带任意参数(任意数目, 任意类型)的函数”。而在C++中,表示“不带参数的函数”。所以在C代码中...

【《jQuery实战》学习笔记02】第二章 创建元素包装集(jQuery对象)

**前言** 昨天终于写了第一篇学习笔记【《jQuery实战》学习笔记一】第一章jQuery引荐,发表了一天获得31个阅读量,虽然没人给我评论,我但感觉已经很不错了!今天把第二章也看完了,于是把笔记整...
  • kessonc
  • kessonc
  • 2014年01月12日 20:09
  • 623

C++编程思想 第二章 对象的创建与使用

1语言的翻译过程 静态类型检查 类型检查(type checking)是编译器在第一遍中完成的。类型检查是检查函数参数是否正确使用,以防止许多程序设计错误。由于类型检查是在编译阶段而不是程序运行阶...

JAVASCRIPT面向对象(创建对象)第二章

待续

C++面向对象第二章

  • 2012年05月12日 18:03
  • 285KB
  • 下载

OC基础第二章:面向对象

1、面向对象(object-oriented ;简称: OO) 定义为:按人们认识客观世界的系统思维方式,采用基于对象(实体)的概念建立模型,模拟客观世界分析、设计、实现软件的办法。通过面向对象的...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:第二章 创建和销毁对象
举报原因:
原因补充:

(最多只允许输入30个字)