﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>语源科技BlogJava-分享天空</title><link>http://www.blogjava.net/jwakak/</link><description>共同进步</description><language>zh-cn</language><lastBuildDate>Sun, 12 Apr 2026 13:24:43 GMT</lastBuildDate><pubDate>Sun, 12 Apr 2026 13:24:43 GMT</pubDate><ttl>60</ttl><item><title>SPRING设计思想之工厂模式 </title><link>http://www.blogjava.net/jwakak/archive/2007/11/12/159879.html</link><dc:creator>麻辣教师</dc:creator><author>麻辣教师</author><pubDate>Mon, 12 Nov 2007 01:47:00 GMT</pubDate><guid>http://www.blogjava.net/jwakak/archive/2007/11/12/159879.html</guid><wfw:comment>http://www.blogjava.net/jwakak/comments/159879.html</wfw:comment><comments>http://www.blogjava.net/jwakak/archive/2007/11/12/159879.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/jwakak/comments/commentRss/159879.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/jwakak/services/trackbacks/159879.html</trackback:ping><description><![CDATA[<div><a id="viewpost1_TitleUrl" href="http://www.blogjava.net/tij/archive/2007/11/11/159696.html">SPRING设计思想之工厂模式</a> </div>
<font style="background-color: #cce8cf">工厂模式：根据调用数据返回某个类的一个实例，此类可以是多个类的某一个类，通常，这些类满足共同的规则或父类。调用者只关心：是否满足某种规范和是否可供自己正常调用。<br />
厂模式定义:提供创建对象的接口.
<p><strong><em>为何使用?</em></strong><br />
工厂模式是我们最常用的模式了,著名的Jive论坛 ,就大量使用了工厂模式，工厂模式在Java程序系统可以说是随处可见。</p>
<p>为什么工厂模式是如此常用？因为工厂模式就相当于创建实例对象的new，我们经常要根据类Class生成实例对象，如A a=new A() 工厂模式也是用来创建实例对象的，所以以后new时就要多个心眼，是否可以考虑实用工厂模式，虽然这样做，可能多做一些工作，但会给你系统带来更大的可扩展性和尽量少的修改量。</p>
<p>我们以类Sample为例， 如果我们要创建Sample的实例对象:</p>
<p>Sample sample=new Sample();</p>
<p>可是，实际情况是，通常我们都要在创建sample实例时做点初始化的工作,比如赋值 查询数据库等。</p>
<p>首先，我们想到的是，可以使用Sample的构造函数，这样生成实例就写成:</p>
<p>Sample sample=new Sample(参数);</p>
<p>但是，如果创建sample实例时所做的初始化工作不是象赋值这样简单的事，可能是很长一段代码，如果也写入构造函数中，那你的代码很难看了（就需要Refactor重整）。</p>
<p>为什么说代码很难看，初学者可能没有这种感觉，我们分析如下，初始化工作如果是很长一段代码，说明要做的工作很多，将很多工作装入一个方法中，相当于将很多鸡蛋放在一个篮子里，是很危险的，这也是有背于Java面向对象的原则，面向对象的封装(Encapsulation)和分派(Delegation)告诉我们，尽量将长的代码分派&#8220;切割&#8221;成每段，将每段再&#8220;封装&#8221;起来(减少段和段之间偶合联系性)，这样，就会将风险分散，以后如果需要修改，只要更改每段，不会再发生牵一动百的事情。</p>
<p>在本例中，首先，我们需要将创建实例的工作与使用实例的工作分开, 也就是说，让创建实例所需要的大量初始化工作从Sample的构造函数中分离出去。</p>
<p>这时我们就需要Factory工厂模式来生成对象了，不能再用上面简单new Sample(参数)。还有,如果Sample有个继承如MySample, 按照面向接口编程,我们需要将Sample抽象成一个接口.现在Sample是接口,有两个子类MySample 和HisSample .我们要实例化他们时,如下:</p>
<p>Sample mysample=new MySample();<br />
Sample hissample=new HisSample();</p>
<p>随着项目的深入,Sample可能还会"生出很多儿子出来", 那么我们要对这些儿子一个个实例化,更糟糕的是,可能还要对以前的代码进行修改:加入后来生出儿子的实例.这在传统程序中是无法避免的.</p>
<p>但如果你一开始就有意识使用了工厂模式,这些麻烦就没有了.</p>
<p><strong>工厂方法<br />
</strong>你会建立一个专门生产Sample实例的工厂:</p>
<table style="width: 80%" cellspacing="3" cellpadding="0" width="80%" border="0">
    <tbody>
        <tr>
            <td style="padding-right: 2.25pt; padding-left: 2.25pt; background: #cccccc; padding-bottom: 2.25pt; padding-top: 2.25pt">
            <p>public class Factory{</p>
            <p>　　public static Sample creator(int which){</p>
            <p>　　//getClass 产生Sample 一般可使用动态类装载装入类。<br />
            　　if (which==1)<br />
            　　　　return new SampleA();<br />
            　　else if (which==2)<br />
            　　　　return new SampleB();</p>
            <p>　　}</p>
            <p>}</p>
            </td>
        </tr>
    </tbody>
</table>
<p>那么在你的程序中,如果要实例化Sample时.就使用</p>
<p>Sample sampleA=Factory.creator(1);</p>
<p>这样,在整个就不涉及到Sample的具体子类,达到封装效果,也就减少错误修改的机会,这个原理可以用很通俗的话来比喻:就是具体事情做得越多,越容易范错误.这每个做过具体工作的人都深有体会,相反,官做得越高,说出的话越抽象越笼统,范错误可能性就越少.好象我们从编程序中也能悟出人生道理?呵呵.</p>
<p>使用工厂方法 要注意几个角色，首先你要定义产品接口，如上面的Sample,产品接口下有Sample接口的实现类,如SampleA,其次要有一个factory类，用来生成产品Sample，如下图，最右边是生产的对象Sample：</p>
<p>&nbsp;</p>
<p>进一步稍微复杂一点，就是在工厂类上进行拓展，工厂类也有继承它的实现类concreteFactory了<strong><em>。</em></strong></p>
<p><strong>抽象工厂</strong><br />
工厂模式中有: 工厂方法(Factory Method) 抽象工厂(Abstract Factory).</p>
<p style="margin-bottom: 12pt">这两个模式区别在于需要创建对象的复杂程度上。如果我们创建对象的方法变得复杂了,如上面工厂方法中是创建一个对象Sample,如果我们还有新的产品接口Sample2.</p>
<p style="margin-bottom: 12pt">这里假设：Sample有两个concrete类SampleA和SamleB，而Sample2也有两个concrete类Sample2A和SampleB2</p>
<p style="margin-bottom: 12pt">那么，我们就将上例中Factory变成抽象类,将共同部分封装在抽象类中,不同部分使用子类实现，下面就是将上例中的Factory拓展成抽象工厂:</p>
<table style="width: 80%" cellspacing="3" cellpadding="0" width="100%" border="0">
    <tbody>
        <tr>
            <td style="padding-right: 2.25pt; padding-left: 2.25pt; background: #cccccc; padding-bottom: 2.25pt; padding-top: 2.25pt">
            <p>public abstract class Factory{</p>
            <p>　　public abstract Sample creator();</p>
            <p>　　public abstract Sample2 creator(String name); </p>
            <p>}</p>
            <p>public class SimpleFactory extends Factory{</p>
            <p>　　public Sample creator(){<br />
            　　　　.........<br />
            　　　　return new SampleA<br />
            　　}</p>
            <p>　　public Sample2 creator(String name){<br />
            　　　　.........<br />
            　　　　return new Sample2A<br />
            　　}</p>
            <p>}</p>
            <p>public class BombFactory extends Factory{</p>
            <p>　　public Sample creator(){<br />
            　　　　......<br />
            　　　　return new SampleB <br />
            　　}</p>
            <p>　　public Sample2 creator(String name){<br />
            　　　　......<br />
            　　　　return new Sample2B<br />
            　　}</p>
            <p>}</p>
            </td>
        </tr>
    </tbody>
</table>
<p>从上面看到两个工厂各自生产出一套Sample和Sample2,也许你会疑问，为什么我不可以使用两个工厂方法来分别生产Sample和Sample2? </p>
<p>抽象工厂还有另外一个关键要点，是因为 SimpleFactory内，生产Sample和生产Sample2的方法之间有一定联系，所以才要将这两个方法捆绑在一个类中，这个工厂类有其本身特征，也许制造过程是统一的，比如：制造工艺比较简单，所以名称叫SimpleFactory。<br />
</p>
<p>在实际应用中，工厂方法用得比较多一些，而且是和动态类装入器组合在一起应用，</p>
<p><strong>举例</strong></p>
<p>我们以Jive的ForumFactory为例，这个例子在前面的Singleton模式中我们讨论过，现在再讨论其工厂模式:</p>
<table style="width: 97%" cellspacing="3" cellpadding="0" width="97%" border="0">
    <tbody>
        <tr>
            <td style="padding-right: 2.25pt; padding-left: 2.25pt; background: #cccccc; padding-bottom: 2.25pt; padding-top: 2.25pt">
            <p>public abstract class ForumFactory {</p>
            <p>　　private static Object initLock = new Object();<br />
            　　private static String className = "com.jivesoftware.forum.database.DbForumFactory";<br />
            　　private static ForumFactory factory = null; </p>
            <p>　　public static ForumFactory getInstance(Authorization authorization) {<br />
            　　　　//If no valid authorization passed in, return null.<br />
            　　　　if (authorization == null) {<br />
            　　　　　　return null;<br />
            　　　　}<br />
            　　　　//以下使用了Singleton 单态模式<br />
            　　　　if (factory == null) {<br />
            　　　　　　synchronized(initLock) {<br />
            　　　　　　　　if (factory == null) {<br />
            　　　　　　　　　　　　...... </p>
            <p>　　　　　　　　　　try {<br />
            　　　　　　　　　　　　　　//动态转载类<br />
            　　　　　　　　　　　　　　Class c = Class.forName(className);<br />
            　　　　　　　　　　　　　　factory = (ForumFactory)c.newInstance();<br />
            　　　　　　　　　　}<br />
            　　　　　　　　　　catch (Exception e) {<br />
            　　　　　　　　　　　　　　return null;<br />
            　　　　　　　　　　}<br />
            　　　　　　　　}<br />
            　　　　　　}<br />
            　　　　}</p>
            <p>　　　　//Now, 返回 proxy.用来限制授权对forum的访问<br />
            　　　　return new ForumFactoryProxy(authorization, factory,<br />
            　　　　　　　　　　　　　　　　　　　　factory.getPermissions(authorization));<br />
            　　}</p>
            <p>　　//真正创建forum的方法由继承forumfactory的子类去完成.<br />
            　　public abstract Forum createForum(String name, String description)<br />
            　　throws UnauthorizedException, ForumAlreadyExistsException;</p>
            <p>　　....</p>
            <p>}</p>
            </td>
        </tr>
    </tbody>
</table>
<p>因为现在的Jive是通过数据库系统存放论坛帖子等内容数据,如果希望更改为通过文件系统实现,这个工厂方法ForumFactory就提供了提供动态接口:</p>
<p>private static String className = "com.jivesoftware.forum.database.DbForumFactory";</p>
<p>你可以使用自己开发的创建forum的方法代替com.jivesoftware.forum.database.DbForumFactory就可以.</p>
<p>在上面的一段代码中一共用了三种模式,除了工厂模式外,还有Singleton单态模式,以及proxy模式,proxy模式主要用来授权用户对forum的访问,因为访问forum有两种人:一个是注册用户 一个是游客guest,那么那么相应的权限就不一样,而且这个权限是贯穿整个系统的,因此建立一个proxy,类似网关的概念,可以很好的达到这个效果. &nbsp;</p>
<p>看看Java宠物店中的CatalogDAOFactory:</p>
<table cellspacing="0" cellpadding="0" width="100%" bgcolor="#cccccc" border="0">
    <tbody>
        <tr>
            <td>public class CatalogDAOFactory {
            <p>　　/**</p>
            <p>　　* 本方法制定一个特别的子类来实现DAO模式。<br />
            　　* 具体子类定义是在J2EE的部署描述器中。<br />
            　　*/</p>
            <p>　　public static CatalogDAO getDAO() throws CatalogDAOSysException {</p>
            <p>　　　　CatalogDAO catDao = null;</p>
            <p>　　　　try {</p>
            <p>　　　　　　InitialContext ic = new InitialContext();<br />
            　　　　　　//动态装入CATALOG_DAO_CLASS<br />
            　　　　　　//可以定义自己的CATALOG_DAO_CLASS，从而在无需变更太多代码<br />
            　　　　　　//的前提下，完成系统的巨大变更。</p>
            <p>　　　　　　String className =(String) ic.lookup(JNDINames.CATALOG_DAO_CLASS);</p>
            <p>　　　　　　catDao = (CatalogDAO) Class.forName(className).newInstance();</p>
            <p>　　　　} catch (NamingException ne) {</p>
            <p>　　　　　　throw new CatalogDAOSysException("<br />
            　　　　　　　　CatalogDAOFactory.getDAO: NamingException while <br />
            　　　　　　　　　　getting DAO type : "n" + ne.getMessage());</p>
            <p>　　　　} catch (Exception se) {</p>
            <p>　　　　　　throw new CatalogDAOSysException("<br />
            　　　　　　　　CatalogDAOFactory.getDAO: Exception while getting <br />
            　　　　　　　　　　DAO type : "n" + se.getMessage());</p>
            <p>　　　　}</p>
            <p>　　　　return catDao;</p>
            <p>　　}</p>
            <p>}<br />
            </p>
            </td>
        </tr>
    </tbody>
</table>
<p><br clear="all" />
</p>
<p>CatalogDAOFactory是典型的工厂方法，catDao是通过动态类装入器className获得CatalogDAOFactory具体实现子类，这个实现子类在Java宠物店是用来操作catalog数据库，用户可以根据数据库的类型不同，定制自己的具体实现子类，将自己的子类名给与CATALOG_DAO_CLASS变量就可以。</p>
<p>由此可见，工厂方法确实为系统结构提供了非常灵活强大的动态扩展机制，只要我们更换一下具体的工厂方法，系统其他地方无需一点变换，就有可能将系统功能进行改头换面的变化。</p>
<p>设计模式如何在具体项目中应用见<a href="http://www.jdon.com/mybook/index.htm" target="_blank">《Java实用系统开发指南》</a></font></p>
<img src ="http://www.blogjava.net/jwakak/aggbug/159879.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/jwakak/" target="_blank">麻辣教师</a> 2007-11-12 09:47 <a href="http://www.blogjava.net/jwakak/archive/2007/11/12/159879.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>转贴：新手东西－－－自己看的</title><link>http://www.blogjava.net/jwakak/archive/2007/11/12/159873.html</link><dc:creator>麻辣教师</dc:creator><author>麻辣教师</author><pubDate>Mon, 12 Nov 2007 01:27:00 GMT</pubDate><guid>http://www.blogjava.net/jwakak/archive/2007/11/12/159873.html</guid><wfw:comment>http://www.blogjava.net/jwakak/comments/159873.html</wfw:comment><comments>http://www.blogjava.net/jwakak/archive/2007/11/12/159873.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/jwakak/comments/commentRss/159873.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/jwakak/services/trackbacks/159873.html</trackback:ping><description><![CDATA[&nbsp;&nbsp;&nbsp;&nbsp;只有注册用户登录后才能阅读该文。<a href='http://www.blogjava.net/jwakak/archive/2007/11/12/159873.html'>阅读全文</a><img src ="http://www.blogjava.net/jwakak/aggbug/159873.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/jwakak/" target="_blank">麻辣教师</a> 2007-11-12 09:27 <a href="http://www.blogjava.net/jwakak/archive/2007/11/12/159873.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>