詳解MyBatisMapper代理實(shí)現(xiàn)數(shù)據(jù)庫調(diào)用原理

1. Mapper 代理層執(zhí)行

創(chuàng)新互聯(lián)堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:做網(wǎng)站、成都做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的沙灣網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

Mapper 代理上執(zhí)行方法調(diào)用時(shí),調(diào)用被委派給 MapperProxy 來處理。

public class MapperProxy<T> implements InvocationHandler, Serializable {
 private final SqlSession sqlSession;
 private final Class<T> mapperInterface;
 private final Map<Method, MapperMethod> methodCache;

 public MapperProxy(SqlSession sqlSession, Class<T> mapperInterface, Map<Method, MapperMethod> methodCache) {
  this.sqlSession = sqlSession;
  this.mapperInterface = mapperInterface;
  this.methodCache = methodCache;
 }

 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  if (Object.class.equals(method.getDeclaringClass())) {
   try {
    return method.invoke(this, args);
   } catch (Throwable t) {
    throw ExceptionUtil.unwrapThrowable(t);
   }
  }
  // 接口里聲明的方法,轉(zhuǎn)換為 MapperMethod 來調(diào)用
  final MapperMethod mapperMethod = cachedMapperMethod(method);

  // 與 Spring 集成時(shí)此處的 sqlSession 仍然 SqlSessionTemplate
  return mapperMethod.execute(sqlSession, args);
 }

 private MapperMethod cachedMapperMethod(Method method) {
  MapperMethod mapperMethod = methodCache.get(method);
  if (mapperMethod == null) {
   mapperMethod = new MapperMethod(mapperInterface, method, sqlSession.getConfiguration());
   methodCache.put(method, mapperMethod);
  }
  return mapperMethod;
 }
}

MapperMethod 根據(jù) mapperInterface.getName() + "." + method.getName() 從 Configuration 對(duì)象里找到對(duì)應(yīng)的 MappedStatement ,從而得到要執(zhí)行的 SQL 操作類型(insert/delete/update/select/flush),然后調(diào)用傳入的 sqlSession 實(shí)例上的相應(yīng)的方法。

public Object execute(SqlSession sqlSession, Object[] args) {
 Object result;
 if (SqlCommandType.INSERT == command.getType()) {
  // 把參數(shù)轉(zhuǎn)換為 SqlSession 能處理的格式
  Object param = method.convertArgsToSqlCommandParam(args);

  // 在 sqlSession 上執(zhí)行并處理結(jié)果
  result = rowCountResult(sqlSession.insert(command.getName(), param));
 } else if (SqlCommandType.UPDATE == command.getType()) {
  Object param = method.convertArgsToSqlCommandParam(args);
  result = rowCountResult(sqlSession.update(command.getName(), param));
 ...省略

如果上述方法傳入的是 SqlSessionTemplate ,那么這些方法調(diào)用會(huì)被 SqlSessionInterceptor 攔截,加入與 Spring 事務(wù)管理機(jī)制協(xié)作的邏輯,具體可以看這篇文章MyBatis 事務(wù)管理,這里不再展開,最終會(huì)調(diào)用到 DefaultSqlSession 實(shí)例上的方法。

2. 會(huì)話層的執(zhí)行過程

SqlSession 里聲明的所有方法的第一個(gè)參數(shù)如果是 String statement ,則都是 mapperInterface.getName() + "." + method.getName() ,表示要調(diào)用的 SQL 語句的標(biāo)識(shí)符。通過它從 configuration 找到 MappedStatement 。

會(huì)話層最主要的邏輯是進(jìn)行參數(shù)的包裝,獲取對(duì)應(yīng)的 MappedStatement ,然后調(diào)用持有的 Executor 的方法去執(zhí)行。

public class DefaultSqlSession implements SqlSession {
 private Configuration configuration;
 private Executor executor;

 private boolean autoCommit;
 private boolean dirty;
 private List<Cursor<?>> cursorList;

 public DefaultSqlSession(Configuration configuration, Executor executor, boolean autoCommit) {
  this.configuration = configuration;
  this.executor = executor;
  this.dirty = false;
  this.autoCommit = autoCommit;
 }

 public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {
  try {
   MappedStatement ms = configuration.getMappedStatement(statement);
   return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);
  } catch (Exception e) {
   throw ExceptionFactory.wrapException("Error querying database. Cause: " + e, e);
  } finally {
   ErrorContext.instance().reset();
  }
 }

3. Executor 執(zhí)行的過程

我們知道 JDBC 里有三種數(shù)據(jù)庫語句: java.sql.Statement/PreparedStatement/CallableStatement ,每種語句的執(zhí)行方式是不一樣的,MyBatis 創(chuàng)建了 StatementHandler 抽象來表示數(shù)據(jù)庫語句的處理邏輯,有對(duì)應(yīng)的三種具體實(shí)現(xiàn): SimpleStatementHandler/PreparedStatementHandler/CallableStatementHandler 。

RoutingStatementHandler 是個(gè)門面模式,構(gòu)建時(shí)根據(jù)要執(zhí)行的數(shù)據(jù)庫語句類型實(shí)例化 SimpleStatementHandler/PreparedStatementHandler/CallableStatementHandler 中的一個(gè)類作為目標(biāo) delegate,并把調(diào)用都轉(zhuǎn)給這個(gè) delegate 的方法。

不同的 handler 實(shí)現(xiàn)實(shí)現(xiàn)了對(duì)應(yīng)的:數(shù)據(jù)庫語句的創(chuàng)建、參數(shù)化設(shè)置、執(zhí)行語句。

通過這層抽象,MyBatis 統(tǒng)一了 Executor 里的執(zhí)行流程,以下以 SimpleExecutor 的流程為例:

1. 對(duì)于傳入的 MappedStatement ms ,得到 Configuration configuration 。

2. configuration 通過 ms 的語句類型得到一個(gè) RoutingStatementHandler 的實(shí)例(內(nèi)部有個(gè) delegate 可以委派) handler ,并用 InterceptorChain 對(duì) handler 實(shí)例進(jìn)行裝飾。

3. 通過 SimpleExecutor 持有的 Transaction 實(shí)例獲取對(duì)應(yīng)的數(shù)據(jù)庫連接 connection。

4. handler 通過數(shù)據(jù)庫連接初始化數(shù)據(jù)庫語句 java.sql.Statement 或其子類 stmt ,設(shè)置超時(shí)時(shí)間和 fetchSize 。

5. 用 handler 對(duì) stmt 進(jìn)行參數(shù)化處理(比如 PreparedStatement 設(shè)置預(yù)編譯語句的參數(shù)值)。

6. handler 執(zhí)行相應(yīng)的 stmt 完成數(shù)據(jù)庫操作。

7. 用 ResultSetHandler 對(duì)結(jié)果集進(jìn)行處理。 ResultSetHandler 會(huì)調(diào)用 TypeHandler 來進(jìn)行 Java 類型與數(shù)據(jù)庫列類型之間轉(zhuǎn)換。

// SimpleExecutor
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();

  // 創(chuàng)建 handler 來負(fù)責(zé)具體的執(zhí)行
  StatementHandler handler = configuration.newStatementHandler(wrapper, ms, parameter, rowBounds, resultHandler, boundSql);

  // 創(chuàng)建數(shù)據(jù)庫語句
  stmt = prepareStatement(handler, ms.getStatementLog());

  // 執(zhí)行數(shù)據(jù)庫操作
  return handler.<E>query(stmt, resultHandler);
 } finally {
  closeStatement(stmt);
 }
}

// Configuration
public StatementHandler newStatementHandler(Executor executor, MappedStatement mappedStatement, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
 StatementHandler statementHandler = new RoutingStatementHandler(executor, mappedStatement, parameterObject, rowBounds, resultHandler, boundSql);
 statementHandler = (StatementHandler) interceptorChain.pluginAll(statementHandler);
 return statementHandler;
}

// RoutingStatementHandler
public RoutingStatementHandler(Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
 // 根據(jù)SQL語句的執(zhí)行方式創(chuàng)建對(duì)應(yīng)的 handler 實(shí)例
 switch (ms.getStatementType()) {
  case STATEMENT:
  delegate = new SimpleStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
  break;
  case PREPARED:
  delegate = new PreparedStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
  break;
  case CALLABLE:
  delegate = new CallableStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
  break;
  default:
  throw new ExecutorException("Unknown statement type: " + ms.getStatementType());
 }
}

private Statement prepareStatement(StatementHandler handler, Log statementLog) throws SQLException {
 // 創(chuàng)建數(shù)據(jù)庫連接
 Connection connection = getConnection(statementLog);

 // 創(chuàng)建數(shù)據(jù)庫語句
 Statement stmt = handler.prepare(connection, transaction.getTimeout());

 // 參數(shù)化設(shè)置
 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;
 }
}

// BaseStatementHandler
public Statement prepare(Connection connection, Integer transactionTimeout) throws SQLException {
 ErrorContext.instance().sql(boundSql.getSql());
 Statement statement = null;
 try {
  // 由具體的子類來創(chuàng)建對(duì)應(yīng)的 Statement 實(shí)例
  statement = instantiateStatement(connection);

  // 通用參數(shù)設(shè)置
  setStatementTimeout(statement, transactionTimeout);
  setFetchSize(statement);
  return statement;
 } catch (SQLException e) {
  closeStatement(statement);
  throw e;
 } catch (Exception e) {
  closeStatement(statement);
  throw new ExecutorException("Error preparing statement. Cause: " + e, e);
 }
}

// PreparedStatementHandler
protected Statement instantiateStatement(Connection connection) throws SQLException {
 String sql = boundSql.getSql();
 if (mappedStatement.getKeyGenerator() instanceof Jdbc3KeyGenerator) {
  String[] keyColumnNames = mappedStatement.getKeyColumns();
  if (keyColumnNames == null) {
   return connection.prepareStatement(sql, PreparedStatement.RETURN_GENERATED_KEYS);
  } else {
   return connection.prepareStatement(sql, keyColumnNames);
  }
 } else if (mappedStatement.getResultSetType() != null) {
  return connection.prepareStatement(sql, mappedStatement.getResultSetType().getValue(), ResultSet.CONCUR_READ_ONLY);
 } else {
  return connection.prepareStatement(sql);
 }
}

// PreparedStatementHandler
public void parameterize(Statement statement) throws SQLException {
 parameterHandler.setParameters((PreparedStatement) statement);
}

4. 問題

只在 XML 里定義 SQL、沒有對(duì)應(yīng)的 Java 接口類能否使用 MyBatis ?

答:可以,通過 SqlSession 的方法來調(diào)用 XML 里的 SQL 語句。

Mapper 接口類里可以進(jìn)行方法重載嗎?

答:不能,因?yàn)?MyBatis 里根據(jù) 類名 + “.” + 方法名 來查找 SQL 語句,重載會(huì)導(dǎo)致這樣的組合出現(xiàn)多條結(jié)果。

以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。

分享題目:詳解MyBatisMapper代理實(shí)現(xiàn)數(shù)據(jù)庫調(diào)用原理
URL地址:http://muchs.cn/article28/pisejp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站維護(hù)網(wǎng)站內(nèi)鏈、做網(wǎng)站微信小程序、企業(yè)建站、云服務(wù)器

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

h5響應(yīng)式網(wǎng)站建設(shè)