溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Mybaits如何實(shí)現(xiàn)打印sql語句

發(fā)布時間:2020-07-16 14:26:55 來源:億速云 閱讀:195 作者:小豬 欄目:開發(fā)技術(shù)

這篇文章主要為大家展示了Mybaits如何實(shí)現(xiàn)打印sql語句,內(nèi)容簡而易懂,下面讓小編帶大家一起學(xué)習(xí)一下吧。

mybatis本身沒有提供日志的實(shí)現(xiàn),引入的是第三方組件。mybatis支持多個第三方日志插件,優(yōu)先級由低到高為slf4J、commonsLoging、Log4J2、Log4J和JdkLog。

mybatis中有一個LogFactory,獲取log的工廠類,在工程類中可以回去對應(yīng)的日志實(shí)現(xiàn)。分析工程類,可以發(fā)現(xiàn)mybatis如何來選擇log

public static Log getLog(String logger) {
 try {
 return logConstructor.newInstance(logger);
 } catch (Throwable t) {
 throw new LogException("Error creating logger for logger " + logger + ". Cause: " + t, t);
 }
}

關(guān)于logConstructor的加載如下

static {
 tryImplementation(LogFactory::useSlf4jLogging);
 tryImplementation(LogFactory::useCommonsLogging);
 tryImplementation(LogFactory::useLog4J2Logging);
 tryImplementation(LogFactory::useLog4JLogging);
 tryImplementation(LogFactory::useJdkLogging);
 tryImplementation(LogFactory::useNoLogging);
}
private static void tryImplementation(Runnable runnable) {
 if (logConstructor == null) {
 try {
  runnable.run();
 } catch (Throwable t) {
  // ignore
 }
 }
}

在 tryImplementation ,中會設(shè)置mybatis使用的log類型。把引用的log設(shè)置到logConstructor中后,后續(xù)其他類型的log也不會再加載。所以在mybatis中優(yōu)先級由低到高為slf4J、commonsLoging、Log4J2、Log4J和JdkLog。感覺也是屬于SPI的一種實(shí)現(xiàn)方式,不同的是各種類型的第三方日志,無法形成一個統(tǒng)一的接口。故此,mybatis為了解決這一問題,使用了適配器模式。

Mybaits如何實(shí)現(xiàn)打印sql語句

適配器的實(shí)現(xiàn)一般是讓適配器實(shí)現(xiàn)或者繼承目標(biāo),并且內(nèi)部持有一個適配者的引用。這樣調(diào)用目標(biāo)對象方法,實(shí)際上是調(diào)用適配者的方法。

mybatis 又是如何把這log,用起來的。根據(jù)mybatis的習(xí)慣,應(yīng)該會使用代理模式,來打印這個日志。 舉例查詢的語句查看,根據(jù)MapperProxy,查到最后查詢的語句

public <E> List<E> doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) throws SQLException {
 Statement stmt = null;
 try {
 Configuration configuration = ms.getConfiguration();
 StatementHandler handler = configuration.newStatementHandler(wrapper, ms, parameter, rowBounds, resultHandler, boundSql);
 stmt = prepareStatement(handler, ms.getStatementLog());
 return handler.query(stmt, resultHandler);
 } finally {
 closeStatement(stmt);
 }
}
private Statement prepareStatement(StatementHandler handler, Log statementLog) throws SQLException {
 Statement stmt;
 Connection connection = getConnection(statementLog);
 stmt = handler.prepare(connection, transaction.getTimeout());
 handler.parameterize(stmt);
 return stmt;
}
protected Connection getConnection(Log statementLog) throws SQLException {
 Connection connection = transaction.getConnection();
 if (statementLog.isDebugEnabled()) {
 return ConnectionLogger.newInstance(connection, statementLog, queryStack);
 } else {
 return connection;
 }
}

到此處可以看到mybatis在獲取連接的時候,會根據(jù)日志的打印級別來判斷是否會創(chuàng)建一個代理類。到這里就基本可以猜到,在代理類中,mybatis會去打印這個sql的語句

public static Connection newInstance(Connection conn, Log statementLog, int queryStack) {
 InvocationHandler handler = new ConnectionLogger(conn, statementLog, queryStack);
 ClassLoader cl = Connection.class.getClassLoader();
 return (Connection) Proxy.newProxyInstance(cl, new Class[]{Connection.class}, handler);
}

用 ConnectionLogger 來舉例,看到里面的invoke的方法

public Object invoke(Object proxy, Method method, Object[] params)
 throws Throwable {
 try {
 if (Object.class.equals(method.getDeclaringClass())) {
  return method.invoke(this, params);
 }
 if ("prepareStatement".equals(method.getName())) {
  if (isDebugEnabled()) {
  debug(" Preparing: " + removeBreakingWhitespace((String) params[0]), true);
  }
  PreparedStatement stmt = (PreparedStatement) method.invoke(connection, params);
  stmt = PreparedStatementLogger.newInstance(stmt, statementLog, queryStack);
  return stmt;
 } else if ("prepareCall".equals(method.getName())) {
  if (isDebugEnabled()) {
  debug(" Preparing: " + removeBreakingWhitespace((String) params[0]), true);
  }
  PreparedStatement stmt = (PreparedStatement) method.invoke(connection, params);
  stmt = PreparedStatementLogger.newInstance(stmt, statementLog, queryStack);
  return stmt;
 } else if ("createStatement".equals(method.getName())) {
  Statement stmt = (Statement) method.invoke(connection, params);
  stmt = StatementLogger.newInstance(stmt, statementLog, queryStack);
  return stmt;
 } else {
  return method.invoke(connection, params);
 }
 } catch (Throwable t) {
 throw ExceptionUtil.unwrapThrowable(t);
 }
}

可以看到,mybatis在里面還可以更具情況創(chuàng)建代理類。代理類又一次被代理,這也是mybatis喜歡的編程方式,比如插件也是代理類再次被代理,來實(shí)現(xiàn)多個插件并行。

以上就是關(guān)于Mybaits如何實(shí)現(xiàn)打印sql語句的內(nèi)容,如果你們有學(xué)習(xí)到知識或者技能,可以把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI