一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|編程技術|正則表達式|

服務器之家 - 編程語言 - JAVA教程 - Mybaits 實現打印sql語句的代碼

Mybaits 實現打印sql語句的代碼

2020-07-16 12:08上學威龍7 JAVA教程

這篇文章主要介紹了Mybaits 實現打印sql語句的代碼,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下

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

mybatis中有一個LogFactory,獲取log的工廠類,在工程類中可以回去對應的日志實現。分析工程類,可以發現mybatis如何來選擇log

?
1
2
3
4
5
6
7
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);
 }
}

關于logConstructor的加載如下

?
1
2
3
4
5
6
7
8
static {
 tryImplementation(LogFactory::useSlf4jLogging);
 tryImplementation(LogFactory::useCommonsLogging);
 tryImplementation(LogFactory::useLog4J2Logging);
 tryImplementation(LogFactory::useLog4JLogging);
 tryImplementation(LogFactory::useJdkLogging);
 tryImplementation(LogFactory::useNoLogging);
}
?
1
2
3
4
5
6
7
8
9
private static void tryImplementation(Runnable runnable) {
 if (logConstructor == null) {
 try {
  runnable.run();
 } catch (Throwable t) {
  // ignore
 }
 }
}

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

Mybaits 實現打印sql語句的代碼

適配器的實現一般是讓適配器實現或者繼承目標,并且內部持有一個適配者的引用。這樣調用目標對象方法,實際上是調用適配者的方法。

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

?
1
2
3
4
5
6
7
8
9
10
11
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);
 }
}
?
1
2
3
4
5
6
7
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;
}
?
1
2
3
4
5
6
7
8
protected Connection getConnection(Log statementLog) throws SQLException {
 Connection connection = transaction.getConnection();
 if (statementLog.isDebugEnabled()) {
 return ConnectionLogger.newInstance(connection, statementLog, queryStack);
 } else {
 return connection;
 }
}

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

?
1
2
3
4
5
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的方法

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
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在里面還可以更具情況創建代理類。代理類又一次被代理,這也是mybatis喜歡的編程方式,比如插件也是代理類再次被代理,來實現多個插件并行。

到此這篇關于Mybaits 實現打印sql語句的代碼的文章就介紹到這了,更多相關Mybaits 打印sql語句內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://juejin.im/post/5f0dbfed6fb9a07ec07b4888

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 俄罗斯毛片免费大全 | 99热色| 精品午夜久久福利大片免费 | 欧美一区二区视频 | 美女视频91| 猫咪av| a及毛片 | a天堂视频| 美国video | 日日操美女 | 免费波多野结衣庭教师 | 男女男精品视频 | 国产欧美另类久久精品91 | 亚洲 欧美 日韩 国产 视频 | 天海翼最新作品 | 久久er99热精品一区二区 | 果冻传媒天美传媒网址入口 | 欧美精品久久久久久久影视 | 亚洲精品成人a | 秋霞午夜伦午夜高清福利片 | 亚洲天堂999| 日韩毛片免费线上观看 | 亚洲国产精品综合欧美 | 午夜影院c绿象 | 亚洲免费二区 | 免费亚洲视频在线观看 | 日产2021免费一二三四区 | 亚洲黄色片免费看 | 国产资源免费 | 免费a视频在线观看 | 国产一线天 | 日本片免费观看一区二区 | 午夜理论片日本中文在线 | 久热人人综合人人九九精品视频 | 175m美女被网友灌醉啪啪玩脚 | 欧美在线成人免费国产 | 国产视频中文字幕 | 好妈妈7在线观看高清 | 日本一区二区三区在线 观看网站 | 精品小视频在线观看 | 国产精品毛片va一区二区三区 |