Hello! 欢迎来到小浪资源网!

如何用CGLIB无侵入式拦截增强java.sql.Statement类?


如何用CGLIB无侵入式拦截增强java.sql.Statement类?

如何使用 cglib 拦截 Java.sql.statement 类

在不修改源代码的情况下拦截增强 java.sql.statement 类,可以使用 cglib。但是,使用 cglib 需要手动使用 enhancer#create() 方法创建一个代理类,手动调用才能触发 callback 的钩子函数,似乎不太方便。

其实,我们可以通过代理 connection 对象来间接拦截 statement 类。修改数据源的 getconnection 方法,使其返回代理对象即可。这样,业务代码无需修改,也能实现拦截增强。

代码示例如下:

立即学习Java免费学习笔记(深入)”;

import net.sf.cglib.proxy.Enhancer; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy;  import javax.sql.DataSource; import java.lang.reflect.Method; import java.sql.Connection; import java.sql.Statement;  public class StatementInterceptor implements MethodInterceptor {      private final DataSource dataSource;      public StatementInterceptor(DataSource dataSource) {         this.dataSource = dataSource;     }      @Override     public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {         if ("createStatement".equals(method.getName())) {             Statement statement = (Statement) proxy.invokeSuper(obj, args);             return new StatementProxy(statement);         }         return proxy.invokeSuper(obj, args);     }      public static void main(String[] args) {         DataSource dataSource = new MyDataSource();         DataSource proxyDataSource = (DataSource) Enhancer.create(dataSource.getClass(), new StatementInterceptor(dataSource));         Connection connection = proxyDataSource.getConnection();         Statement statement = connection.createStatement();         // ...     }      private static class MyDataSource implements DataSource {          @Override         public Connection getConnection() {             return null;         }          @Override         public Connection getConnection(String username, String password) {             return null;         }          @Override         public <T> T unwrap(Class<T> iface) {             return null;         }          @Override         public boolean isWrapperFor(Class<?> iface) {             return false;         }     }      private static class StatementProxy implements Statement {          private final Statement statement;          public StatementProxy(Statement statement) {             this.statement = statement;         }          @Override         public ResultSet executeQuery(String sql) {             // ...             return statement.executeQuery(sql);         }          @Override         public int executeUpdate(String sql) {             // ...             return statement.executeUpdate(sql);         }          // ...     } }

相关阅读