简单工厂模式:Factory

简单工厂模式有什么用途呢?

当创建类较多时,尽量解耦合,

1 工厂方法封装了对象创建的细节,将对象创建代码和其他部分脱离,减少相干性.
2 有利于同类对象创建的统一管理和控制
3 你所关心的仅仅是工厂方法返回的接口方法,不必关心实现细节
 

工厂模式就相当于创建实例对象的new,我们经常要根据类Class生成实例对象,如A a=new A(). 工厂模式也是用来创建实例对

象的,可能多做一些工作,但会给你系统带来更大的可扩展性和尽量少的修改量。

类Sample为例,要创建Sample的实例对象:

Sample sample=new Sample();
可是,实际情况是,通常我们都要在创建sample实例时做点初始化的工作,比如赋值 查询数据库等

首先,我们想到的是,可以使用Sample的构造函数,这样生成实例就写成:

Java codeSample sample=new Sample(参数);

但是,如果创建sample实例时所做的初始化工作不是象赋值这样简单的事,可能是很长一段代码,如果也写入构造函数中,那你

的代码很难看了

初始化工作如果是很长一段代码,说明要做的工作很多,将很多工作装入一个方法中,相当于将很多鸡蛋放在一个篮子里,是很

危险的,这也是有背于Java面向对象的原则,面向对象的封装(Encapsulation)和分派(Delegation)告诉我们,尽量将长的代码

分派“切割”成每段,将每段再“封装 ”起来(减少段和段之间偶合联系性),这样,就会将风险分散,以后如果需要修改,只

要更改每段,不会再发生牵一动百的事情。

我们需要将创建实例的工作与使用实例的工作分开, 也就是说,让创建实例所需要的大量初始化工作从Sample的构造函数中分离

出去。

你想如果有多个类似的类,我们就需要实例化出来多个类。这样代码管理起来就太复杂了。
这个时候你就可以采用工厂方法来封装这个问题。
不能再用上面简单new Sample(参数)。还有,如果Sample有个继承如MySample, 按照面向接口编程,我们需要将Sample抽象成一个

接口.现在Sample是接口,有两个子类MySample 和HisSample

举个更实际的例子,比如你写了个应用,里面用到了数据库的封装,你的应用可以今后需要在不同的数据库环境下运行,可能是

oracle,db2,sql server等,那么连接数据库的代码是不一样的,你用传统的方法,就不得不进行代码修改来适应不同的环境,

非常麻烦,但是如果你采用工厂类的话,将各种可能的数据库连接全部实现在工厂类里面,通过你配置文件的修改来达到连接的

是不同的数据库,那么你今后做迁移的时候代码就不用进行修改了。

我通常都是用xml的配置文件配置许多类型的数据库连接,非常的方便。PS:工厂模式在这方面的使用较多。

 

 我的理解:当需要在程序的很多地方去New出来很多对象时,而每个对象都需要初始化;或者假如以后发现,在代码的某个地方,new出来的对象不对,需要更改时,

假如使用了工厂模式,那么可能之用更改一个地方即可。

举个生活中的例子,假如你买个柜子,如果柜子的生产厂商,把一堆没有组装的、零散的部件卖给你,每个客户购买了以后,都需要回家组装,而这个组装又很麻烦、容易出错,那这就是毛病了。每个客户的动手能力不同,而且容易出错,到最后谁也不买这中柜子了。

所以最好就是每个客户购买的柜子,在出厂时,都组装好。假如以后发现哪种柜子有缺点,只用更改工厂的组装流程即可。

当然,更复杂的问题,如果每个客户都想自己定义柜子的组装方法、尺寸,那就是另外的问题了。

interface Car{
public void run();
public void stop();
}
//具体实现类
class Benz implements Car{
public void run(){
System.out.println("Benz开始启动了。。。。。");
}
public void stop(){
System.out.println("Benz停车了。。。。。");
}
}

class Ford implements Car{
public void run(){
System.out.println("Ford开始启动了。。。");
}
public void stop(){
System.out.println("Ford停车了。。。。");
}
}

class Toyota implements Car{
public void run(){
System.out.println("Toyota开始启动了。。。");
}
public void stop(){
System.out.println("Toyota停车了。。。。");
}
}
//工厂类
class Factory{
public static Car getCarInstance(String type){
Car c=null;
try {
c=(Car)Class.forName("org.jzkangta.factorydemo03."+type).newInstance();
} catch (InstantiationException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

return c;
}
}
//测试类
public class FactoryDemo03 {

public static void main(String[] args) {
Car c=Factory.getCarInstance("Toyota");//****这里调用具体的实现类
if(c!=null){
c.run();
c.stop();
}else{
System.out.println("造不了这种汽车。。。");
}


}

}

//假如你想再增加一个牌子的车,只增加一个具体实现类就OK了(想调用它就在Tooyota那里选择就行了),不用修改任何代码!

posted @ 2010-11-23 09:22  pjh123  阅读(191)  评论(0编辑  收藏  举报