设计模式(七)——门面模式
门面模式(facade pattern),又称外观模式,为子系统中的一组接口提供一个一致的界面, Facade 模式定义了一个高层接口,这个接口使得这一子系统更加容易使用。简而言之,就是把一堆复杂的流程封装成一个接口供给用户更简单的使用,这个设计模式里有三个角色:
1)门面角色( facade ):这是门面模式的核心。它被客户角色调用,因此它熟悉子系统的功能。它内部根据客户角色已有的需求预定了几种功能组合。
2)子系统角色(subsystem):实现了子系统的功能。对它而言, façade 角色就和客户角色一样是未知的,它没有任何 façade 角色的信息和链接。
3)客户角色:调用 façade 角色来完成要得到的功能。
好了,老规矩,假设一个场景,这里说句题外话,在我写这个玩意的时候,每当我假设某个场景的时候,比如什么造汽车,卖水果的场景,我旁边同事就很鄙视的说,不要搞这些没意义的场景,什么卖西瓜造汽车,造你妹啊,叼用都没有的代码,你这误导新人,说的十分激动义愤填膺,我说,我这是为了说明这设计模式的思想,所以我才选一些生活中常用的场景而不引人一些复杂的模型嘛,关键是让别人理解这个思想嘛,要让大家举一反三,我又不是为了写个具体有用的代码让他copy过去直接用的。他说,诡辩,滚~
不过我觉得给他一次机会,所以这次我写个有用点的场景。是这样,门面模式最常见也最有效的一个写法就在jdbc连数据库时,我们知道连接一个数据库,要先加载驱动,获得连接,获得statement,然后执行操作,最后关闭连接,步骤顺序都不能变,如果每次执行都要自己写这些,那真的会蛋蛋疼,所以,我们把这些都封装起来,如下:
先建立我们的门面,但是我们改进下,我们把这个门面写成抽象的,这样扩展性更好:
package com.gy.designpattern.facade; import java.sql.Connection; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement; /** * ClassName:DBFacade
* Function: 抽象的数据库门面.
* Reason: 这样可以使用这个连接上不同类型的数据.
* Date: 2012-7-10 上午10:51:35
* @author gongyong * @version * @since JDK 1.6 * @see */ public abstract class DBFacade { Connection conn = null; Statement stmt = null; ResultSet rs = null; String sqlStr = ""; //获取数据库连接 abstract void getConnection(); //创建statement public void createStatement() { try { stmt = conn.createStatement(); } catch (SQLException e) { e.printStackTrace(); } } //执行查询操作 public void executeQuery(){ try { rs = stmt.executeQuery(sqlStr); } catch (SQLException e) { e.printStackTrace(); } } //封装之前三个必须操作 public void doQuery(String sqlStr){ this.sqlStr = sqlStr; getConnection(); createStatement(); executeQuery(); } public void close(){ try { rs.close(); stmt.close(); conn.close(); } catch (SQLException e) { e.printStackTrace(); } } }
然后我们具体实现这个门面,我们用mysql数据库:
package com.gy.designpattern.facade; import java.sql.DriverManager; /** * ClassName:MysqlUtil
* Function: mysql数据库门面的具体实现.
* Reason: TODO ADD REASON.
* Date: 2012-7-10 上午10:54:46
* @author gongyong * @version * @since JDK 1.6 * @see */ public class MysqlUtil extends DBFacade{ /** * 注入数据库驱动并连接上指定的数据库. * @see com.gy.designpattern.facade.DBFacade#getConnection() */ @Override void getConnection(){ try{ Class.forName("com.mysql.jdbc.Driver"); conn=DriverManager.getConnection("jdbc:mysql://localhost:3306/test","root","123456"); }catch(Exception e){ e.printStackTrace(); } } }
到这里门面已经完成了,看怎么用吧:
package com.gy.designpattern.facade; import java.sql.SQLException; /** * ClassName:Client
* Function: 客户端,用来操作数据库.
* Reason: TODO ADD REASON.
* Date: 2012-7-10 上午10:58:05
* @author gongyong * @version * @since JDK 1.6 * @see */ public class Client { public static void main(String[] args){ //sql语句 String sqlStr = "select * from user"; //先new出一个mysql数据库的操作工具,也就是门面 DBFacade dbFacade=new MysqlUtil(); try { //直接使用该门面中封装好的方法 dbFacade.doQuery(sqlStr); while(dbFacade.rs.next()){ System.out.print(dbFacade.rs.getString(2)); } } catch (SQLException e) { e.printStackTrace(); }finally{ dbFacade.close(); } } }
看吧,是不是封装之后再执行各种数据库的操作就很简单了,客户端完全不需要知道怎么连接数据库等等这些流程,他只需要new一个门面,调用对应的方法就可以了,是不是方面了许多,就是这样。