自定义Druid的拦截器 

Druid是一个JDBC组件,它包括三部分: 

  1. DruidDriver 代理Driver,能够提供基于Filter-Chain模式的插件体系。 
  2. DruidDataSource 高效可管理的数据库连接池。 
  3. SQLParser 

Druid可以做什么? 

1) 可以监控数据库访问性能,Druid内置提供了一个功能强大的StatFilter插件,能够详细统计SQL的执行性能,这对于线上分析数据库访问性能有帮助。 

2) 替换DBCPC3P0。Druid提供了一个高效、功能强大、可扩展性好的数据库连接池。 

3) 数据库密码加密。直接把数据库密码写在配置文件中,这是不好的行为,容易导致安全问题。DruidDruiver和DruidDataSource都支持PasswordCallback。 

4) SQL执行日志,Druid提供了不同的LogFilter,能够支持Common-Logging、Log4j和JdkLog,你可以按需要选择相应的LogFilter,监控你应用的数据库访问情况。 

扩展JDBC,如果你要对JDBC层有编程的需求,可以通过Druid提供的Filter-Chain机制,很方便编写JDBC层的扩展插件。 

-------------------------------------分割线--------------------------------------------

怎么使用Druid提供的Filter-Chain机制,方便编写JDBC层的扩展插件呢?

这才是本篇文章重点要描述的。

https://github.com/alibaba/druid/wiki/%E9%A6%96%E9%A1%B5 druid官方里面好像没找到。

(代码里面的注释真的少得可怜。。。)

不过用过druid的知道里面内置了一些filter

例如在配置


<property name="filters" value="stat,log4j" />


附内置filter的别名

Filter类名

别名

default

com.alibaba.druid.filter.stat.StatFilter

stat

com.alibaba.druid.filter.stat.StatFilter

mergeStat

com.alibaba.druid.filter.stat.MergeStatFilter

encoding

com.alibaba.druid.filter.encoding.EncodingConvertFilter

log4j

com.alibaba.druid.filter.logging.Log4jFilter

log4j2

com.alibaba.druid.filter.logging.Log4j2Filter

slf4j

com.alibaba.druid.filter.logging.Slf4jLogFilter

commonlogging

com.alibaba.druid.filter.logging.CommonsLogFilter

这些Filter都实现了com.alibaba.druid.filter.Filter 这个接口,里面的函数太多。。

从哪里开始?

其实最简单的方法就是仿照已有的Filter来写,例如Log4jFilter,

Log4jFilter 继承 LogFilter,从上面看出,Druid日志插件还支持Slf4jLog,CommonsLog。这里面其实就是根据不同情况打印不同的日志。

LogFilter又继承于FilterEventAdapter这个类,这个类好像很熟悉,貌似Spring里面经常会有xxxAdapter来帮我们对一些接口实现一些默认的方法,而我们只需要继承这个类, 重写里面需要重写方法就可以了。

里面最重要的方法就是


@Override
public boolean preparedStatement_execute(FilterChain chain, PreparedStatementProxy statement) throws SQLException {
    try {
        //抽象方法
        statementExecuteBefore(statement, statement.getSql());

        boolean firstResult = chain.preparedStatement_execute(statement);

        //抽象方法,后面我们会重写这个方法
        this.statementExecuteAfter(statement, statement.getSql(), firstResult);

        return firstResult;

    } catch (SQLException error) {
        //抽象方法
        statement_executeErrorAfter(statement, statement.getSql(), error);
        throw error;
    } catch (RuntimeException error) {
        //抽象方法
        statement_executeErrorAfter(statement, statement.getSql(), error);
        throw error;
    } catch (Error error) {
        //抽象方法
        statement_executeErrorAfter(statement, statement.getSql(), error);
        throw error;
    }

}


好吧,这个方法就是实行FilterChain的preparedStatement_execute方法。这里就是filter的执行链上的一个node把。

依葫芦画瓢

1. 创建一个Filter


/**
 * Created by hzlizhou on 2017/2/5.
 */
public class TestFilter extends FilterEventAdapter{
    @Override
    protected void statementExecuteAfter(StatementProxy statement, String sql, boolean result) {
        System.out.println("haha");
        super.statementExecuteAfter(statement, sql, result);
    }
}


我们自己写一个TestFilter, 继承FilterEventAdapter,重写statementExecuteAfter方法(在SQL语句执行成功以后会执行)

2. 加入到执行链(ChainFilter)

仔细看DruidDataSource,他的父类,里面有个


protected List<Filter> filters  = new CopyOnWriteArrayList<Filter>();
 
<property name="filters" value="stat,log4j" /> DataSource配置的时候就是调用这个set方法
 
public void setFilters(String filters) throws SQLException {
    if (filters != null && filters.startsWith("!")) {
        filters = filters.substring(1);
        this.clearFilters();
    }
    this.addFilters(filters);
}


还有一个set方法是直接加入filters


public void setProxyFilters(List<Filter> filters) {
    if (filters != null) {
        this.filters.addAll(filters);
    }
}


因此,我们在datasource配置中增加


<property name="proxyFilters">
   <list>
      <ref bean="testFilter" />
   </list>
</property>


以及对应的bean


<bean id="testFilter" class="com.netease.urs.druid.filter.TestFilter"></bean>


运行程序,发现每次成功访问数据库都会打印"haha",看来这样做的是对的。这样,我们就可以按照自己的需求自定义一些操作啦(我的目的其实就是数据库同步,把增、删、改的sql语句加入队列中)

原理解析

简单分析下Druid的Filter吧。其实和一般的Filter一样的(Spring Security,Servlet),功能就是在执行真正的业务代码之前和之后增加一些自定义的功能。

顺着刚才的思路往上找,刚才分析到了

@Override
public boolean preparedStatement_execute(FilterChain chain, PreparedStatementProxy statement) throws SQLException {
    try {
        //抽象方法
        statementExecuteBefore(statement, statement.getSql());
      
        //重点是这里!!!!
        boolean firstResult = chain.preparedStatement_execute(statement);

        //抽象方法,后面我们会重写这个方法
        this.statementExecuteAfter(statement, statement.getSql(), firstResult);

        return firstResult;

    } catch (SQLException error) {
        //抽象方法
        statement_executeErrorAfter(statement, statement.getSql(), error);
        throw error;
    } catch (RuntimeException error) {
        //抽象方法
        statement_executeErrorAfter(statement, statement.getSql(), error);
        throw error;
    } catch (Error error) {
        //抽象方法
        statement_executeErrorAfter(statement, statement.getSql(), error);
        throw error;
    }

}

先来看一下FilterChain ,他只有一个实现类FilterChainImpl,有3个成员变量,终于看到了DataSourceProxy(看名字也知道其实就是对DruidDataSource的代理类)


protected int                 pos = 0;
private final DataSourceProxy dataSource;
private final int             filterSize;
 
preparedStatement_executeQuery的方法如下。这样在调用前会把所有的Filter都执行一次其中的preparedStatement_execute。
 
@Override
public boolean preparedStatement_execute(PreparedStatementProxy statement) throws SQLException {
    if (this.pos < filterSize) {
        return nextFilter().preparedStatement_execute(this, statement);
    }
    return statement.getRawObject().execute();
}


preparedStatement_execute在哪里被执行的呢?PreparedStatementProxyImpl#execute()

PreparedStatementProxyImpl#execute()在哪里被执行的呢? DruidPooledPreparedStatement#execute()

(好吧,我承认我是debug看调用栈的。。DruidPooledPreparedStatement#execute()就是iBatis中的调用的了!)

调用的时序大概如下面

druid监控默认密码_抽象方法

DruidPooledPreparedStatement除了execute(),还有executeUpdate(),executeQuery()。但是如果是基于ibatis的sql语句都是使用execute()。