`

一劳永逸的数据库编码解决方案

 
阅读更多

 

一劳永逸的数据库编码解决方案

 

 

问题提出

 

       现在几乎所有的应用系统都无法避免使用数据库系统。在JAVA世界里访问数据库是一件非常轻松的事情,JDBCJAVA应用程序访问数据库提供了一个统一的接口,通过使用JDBC接口开发者无需关心系统最终采用哪种数据库,因为JDBC仅仅是定义了访问几个JAVA的接口类,具体的实现是由数据库厂商提供的,这种做法其实与其他数据库连接方式例如ODBC是类似的。但是在实际的应用过程中,开发者发现离JDBC设计的初衷还是有一定距离,就比如说在存储字符串时的编码问题,我想很多开发者都会遇见这个问题,倒不是因为说解决它有什么技术方面的难度,而是它的的确确非常繁琐。我们必须在每次写入或者读出字符串的时候进行编码和反编码处理;或者说我们可以写一个方法可以进行编码处理的,但又必须在每次数据库操作的时候调用,虽然调用很简单,可是我非得这样吗?要是忘了编码那又要DEBUG了。当然你可能觉得这并没有什么,或者你可能很勤快,喜欢写大量重复的代码,可是你难道没有觉得这种繁琐的工作正在浪费你过于宝贵的青春吗?停止你的键盘输入,让我们来解决这个问题吧!

 

解决思路

 

       在传统的应用程序中数据库操作部分我们可以想象成两层,如图所示:一个是数据库的“连接池”,另外一个业务数据操作层。在这里数据库的连接池是广义的,你可以把JDBC中的DriverManager也当成是连接池,具体的意思就是我们可以通过这层来获取到指定数据库的连接而不去关心它是怎么获取的。如果这个时候数据库系统(有如InformixSQL Server)要求对字符串进行转码才能存储(例如最常见的GBK->ISO8859_1转码),那我们就必须在业务数据操作层来进行,这样有多少业务数据操作我们就要做多少编码转码的工作,太麻烦了,代码中充斥中大量重复的内容。本文提出的解决方案就是利用对获取到的数据库连接实例进行二次封装,也就是在数据库连接池与业务数据操作层之间加入了连接封装层,当然了,我们也完全可以直接将连接封装集成到数据库连接池内部。

 

 

       我们知道进行编码和转码工作都是集中在JDBC的两个接口PreparedStatementResultSet上进行的,主要涉及PreparedStatementsetString方法以及ResultSetgetString方法。前面我们讲过需要加入一个连接封装层来对数据库连接实例进行二次封装,但是怎么通过这个封装来改变PreparedStatementResultSet这两个接口的行为呢?这个问题其实也很简单,因为PreparedStatement接口必须通过Connection接口来获取实例,而ResultSet接口又必须从Statement或者PreparedStatement接口来获取实例,有了这样的级联关系,问题也就迎刃而解了。还是利用我在文章《使用JAVA动态代理实现数据库连接池》中使用的动态接口代理技术。首先我们设计Connection接口的代理类_Connection,这个代理类接管了Connection接口中所有可能获取到Statement或者PreparedStatement接口实例的方法,例如:prepareStatementcreateStatement。改变这两个方法使之返回的是经过接管后的Statement或者PreparedStatement实例。通过对于Statement接口也有相应的代理类_Statement,这个代理类接管用于获取ResultSet接口实例的所有方法,包括对setString方法的接管以决定是否对字符串进行编码处理。对于接口ResultSet的接管类_ResultSet就相应的比较简单,它只需要处理getString方法即可。

 

关键代码

 

       前面我们大概介绍了这个解决方案的思路,下面我们给出关键的实现代码包括Connection的代理类,Statement的代理类,ResultSet的代理类。这些代码是在原来关于数据库连接池实现的基础上进行扩充使之增加对自动编码处理的功能。有需要源码打包的可以通过电子邮件跟我联系。

 

_Connection.java

 

/*

 * Created on 2003-10-23 by Liudong

 */

package lius.pool;

 

import java.sql.*;

 

import java.lang.reflect.*;

 

/**

 * 数据库连接的代理类

 * @author Liudong

 */

class _Connection implements InvocationHandler

{

    private Connection conn = null;

    private boolean coding = false;//指定是否进行字符串转码操作

   

    _Connection(Connection conn, boolean coding){

       this.conn = conn;

       this.coding = coding;

       initConnectionParam(this.conn);

    }

   

    /**

     * Returns the conn.

     * @return Connection

     */

    public Connection getConnection() {

       Class[] interfaces = conn.getClass().getInterfaces();

       if(interfaces==null||interfaces.length==0){

           interfaces = new Class[1];

           interfaces[0] = Connection.class;

       }

       Connection conn2 = (Connection)Proxy.newProxyInstance(

           conn.getClass().getClassLoader(),

           interfaces,this);

       return conn2;

    }

 

    /**

     * @see java.lang.reflect.InvocationHandler#invoke

     */

    public Object invoke(Object proxy, Method m, Object[] args)

       throws Throwable {

       String method = m.getName();

       //调用相应的操作

       Object obj = null;

       try{

           obj = m.invoke(conn, args);

       //接管用于获取语句句柄实例的方法    if((CS.equals(method)||PS.equals(method))&&coding)

              return new _Statement((Statement)obj,true).getStatement();

       }catch(InvocationTargetException e){

           throw e.getTargetException();

       }     

       return obj;

    }

   

    private final static String PS = "prepareStatement";

    private final static String CS = "createStatement";

   

}

 

_Statement.java

 

/*

 * Created on 2003-10-23 by Liudong

 */

package lius.pool;

 

import java.sql.*;

import java.lang.reflect.*;

 

/**

 * 数据库语句对象实例的代理类

 * @author Liudong

 */

class _Statement implements InvocationHandler

{

    private Statement statement ;   //保存所接管对象的实例

    private boolean decode = false; //指定是否进行字符串转码

   

    public _Statement(Statement stmt,boolean decode) {

       this.statement = stmt;

        this.decode = decode;

    }

    /**

     * 获取一个接管后的对象实例

     * @return

     */

    public Statement getStatement() {

       Class[] interfaces = statement.getClass().getInterfaces();

       if(interfaces==null||interfaces.length==0){

           interfaces = new Class[1];

           interfaces[0] = Statement.class;

       }

       Statement stmt = (Statement)Proxy.newProxyInstance(

           statement.getClass().getClassLoader(),

           interfaces,this);

       return stmt;

    }

    /**

     * 方法接管

     */

    public Object invoke(Object proxy, Method m, Object[] args)

       throws Throwable {

       String method = m.getName();

       //接管setString方法

       if(decode && SETSTRING.equals(method)){

           try{

              String param = (String)args[1];

              if(param!=null)

                  param = new String(param.getBytes(),”8859_1”);

              return m.invoke(statement,new Object[]{args[0],param});

           }catch(InvocationTargetException e){

              throw e.getTargetException();            

           }

       }

       //接管executeQuery方法

       if(decode && EXECUTEQUERY.equals(method)){

           try{

              ResultSet rs = (ResultSet)m.invoke(statement,args);

              return new _ResultSet(rs,decode).getResultSet();

           }catch(InvocationTargetException e){

              throw e.getTargetException();            

           }

       }

       try{

           return m.invoke(statement, args);

       }catch(InvocationTargetException e){

           throw e.getTargetException();

       }

    }

    //两个要接管的方法名

    private final static String SETSTRING = "setString";

    private final static String EXECUTEQUERY = "executeQuery";

}

 

 

_ResultSet.java

 

/*

 * Created on 2003-10-23 by Liudong

 */

package lius.pool;

 

import java.sql.ResultSet;

 

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.InvocationTargetException;

import java.lang.reflect.Method;

import java.lang.reflect.Proxy;

 

/**

 * 数据库结果集的代理类

 * @author Liudong

 */

class _ResultSet implements InvocationHandler

{

    private ResultSet rs = null;

    private boolean decode = false;

   

    public _ResultSet(ResultSet rs,boolean decode) {

       this.rs = rs;

       this.decode = decode;

    }

   

    public ResultSet getResultSet(){

       Class[] interfaces = rs.getClass().getInterfaces();

       if(interfaces==null||interfaces.length==0){

           interfaces = new Class[1];

           interfaces[0] = ResultSet.class;

       }

       ResultSet rs2 = (ResultSet)Proxy.newProxyInstance(

           rs.getClass().getClassLoader(),

           interfaces,this);

       return rs2;

    }

   

    /**

     * 结果getString方法

     */

    public Object invoke(Object proxy, Method m, Object[] args)

        throws Throwable

    {

       String method = m.getName();

       if(decode && GETSTRING.equals(method)){

           try{

              String result = (String)m.invoke(rs,args);

              if(result!=null)

                  return new String(result.getBytes(“8859_1”));

              return null;

           }catch(InvocationTargetException e){

              throw e.getTargetException();            

           }

       }

       try{

           return m.invoke(rs, args);

       }catch(InvocationTargetException e){

           throw e.getTargetException();

       }

    }

    private final static String GETSTRING = "getString";

}

 

 

现在我们已经把三个接口的代理类做好了,下一步就是怎么来使用这三个类。其实对于使用者来讲并不需要关心三个类,只需要了解_Connection就可以了,因为另外两个是_Connection直接调用的。为了使用_Connection我们必须传入两个参数,第一个是数据库实际的数据库连接实例,另外一个是布尔值代表是否进行转码处理。我们必须先通过实际的情况获取到数据库连接后再传入_Connection的构造函数作为参数,下面例子告诉你如何来使用_Connection这个类:

 

 

       Connection conn = getConnection();//获取数据库连接

       boolean coding = false;//从配置或者其他地方读取是否进行转码的配置

        //接管数据库连接实例

       _Connection _conn = new _Connection(conn,coding);

       //获得接管后的数据库连接实例,以后直接使用conn2而不是conn

       Connection conn2 = _conn.getConnection();

 

 

因为对一个应用系统来讲,数据库连接的获取必然有统一的方法,在这个方法中加入对连接的接管就可以一劳永逸的解决数据库的编码问题。

 

性能比较

 

       功能没有问题了,开发者接下来就会关心性能的问题,因为在进行一些对响应速度要求很高或者大数据量的处理情况下性能就成为一个非常突出的问题。由于JAVA中的动态接口代理采用的是反射(Reflection)机制,同时又加入我们自己的一些代码例如方法名判断,字符串转码等操作因此在性能上肯定比不上直接使用没有经过接管的数据库连接。但是这点性能上的差别是不是我们可以忍受的呢,为此我做了一个试验对二者进行了比较:

 

测试环境简单描述:

 

使用ACCESS数据库,建两张结构一样的表,计算从获取连接后到插入数据完毕后的时间差,两个程序(直连数据库和使用连接接管)都进行的字符串的转码操作。

 

 

测试结果:

 

插入记录数

直连数据库程序耗时 单位:ms

使用连接接管程序耗时

性能比较

1000

2063

2250

9.0%

5000

8594

8359

-2.7%

10000

16750

17219

2.8%

15000

22187

23000

3.6%

20000

27031

27813

2.9%

 

从上面这张测试结果表中来看,二者的性能的差别非常小,尽管在两万条数据的批量插入的时候时间差别也不会多于一秒钟,这样的结果应该说还是令人满意的,毕竟为了程序良好的结构有时候牺牲一点点性能还是值得的。

 

 

 

分享到:
评论

相关推荐

    ........网络安全解决方案

    ### 网络安全解决方案详解 #### 一、网络信息安全简介 ##### 1.1 安全的概念 安全是指能够识别并消除不安全因素的能力,它包括但不限于最小化资产和资源的漏洞,以及与互联网环境中的风险保持紧密关联。安全不是一...

    用过滤器解决oracle中文乱码问题

    #### 解决方案:设置字符编码过滤器 为了从根本上解决中文乱码问题,我们可以创建一个简单的字符编码过滤器,该过滤器会在每次请求处理之前被调用,用于设置请求的字符编码。下面是一个具体的实现例子: 1. **编写...

    基于协程编程在移动端研发的最佳实践.pptx

    然而,引入协程并非一劳永逸的解决方案,还需要配合全面的监控和性能优化策略。例如,使用Emas等全栈监控工具可以实时监控应用的崩溃、卡顿和内存状况,通过编码规范、代码审查和单元测试来提升代码质量。对于编程...

    cognos 安装

    3. **一劳永逸的解决方案**:为了解决可能的JRE兼容性问题,可以将`cognos\bin\jre\ext`目录下的所有jar包复制到系统的Java路径下。 #### 六、总结 通过以上步骤,您应该能够顺利完成Cognos 10的安装和基本配置。在...

    软件实施专项方案.doc

    1. **需求分析**:在实施软件前,需深入了解业务流程,明确用户需求,以便定制适合企业的解决方案。这涉及到与各部门的沟通,收集功能需求和性能指标。 2. **系统设计**:根据需求分析结果,设计系统的架构、数据库...

    PBKILLER.RAR

    PBKILLER.RAR是一个压缩包,包含了名为"Pbkiller"的工具,专门用于对付PowerBuilder的反编译问题。PowerBuilder是一款强大的...然而,它并不是一劳永逸的解决方案,需要与多种安全措施相结合,才能有效抵御潜在的威胁。

    jsp和servlet操作mysql中文乱码问题的解决办法

    本文将详细解释如何解决JSP和Servlet操作MySQL时出现的中文乱码问题,并提供具体的解决方案。 首先,需要了解中文乱码产生的原因。通常,乱码问题是由编码不一致引起的。当页面、请求、数据库等不同组件的编码设置...

    jasperreport与ireport的配置与使用

    ### JasperReport与iReport的配置与使用 #### 1. JasperReport与iReport简介 ...JasperReport与iReport的组合为开发者提供了强大而灵活的报表解决方案。通过合理配置和使用,可以极大地提高报表开发的效率和质量。

    jasperreportireport中文指南

    针对可能出现的错误,提供解决方案。 ##### 4.2 在Servlet程序中的应用示例 演示如何在Servlet中集成JasperReport来生成报表。 ##### 4.3 动态SQL的示例 展示如何在JasperReport中使用动态SQL查询。 ##### 4.4 ...

    97-things-every-software-architect-should-know

    12. 没有一劳永逸的解决方案:软件架构应该根据项目的具体情况灵活调整。 13. 性能问题应该提前考虑:在项目初期就需要考虑性能问题,而不是等到系统开发完成后再来处理。 14. 应用架构决定应用性能:不同的架构...

    国家标准-软件开发规范

    每一轮测试后都需要有详尽的测试报告,记录测试结果、问题及解决方案。国家标准强调应尽早引入自动化测试工具,提高测试效率。 5. 文档编写:完整的文档是软件生命周期中不可或缺的一部分。国家标准要求编写包括...

    JasperReport 与iReport 的配置与使用

    JasperReport 提供了一套完整的解决方案,包括数据获取、报告设计以及最终报表的渲染和输出。它可以支持多种输出格式,如 PDF、HTML、XML、Excel 和 CSV 等,使得开发者可以根据需求灵活地选择输出方式。 ##### 1.2...

Global site tag (gtag.js) - Google Analytics