执行细节 首先设置了默认的languageDriver
org/mybatis/mybatis/3.5.11/mybatis-3.5.11-sources.jar!/org/apache/ibatis/session/Configuration.java:215
在configuration
的构造方法里
1 languageRegistry.setDefaultDriverClass(XMLLanguageDriver.class);
而在org.apache.ibatis.builder.xml.XMLStatementBuilder#parseStatementNode
中,创建了sqlSource
,这里就会根据前面的 LanguageDriver
的实现选择对应的 sqlSource
,
1 SqlSource sqlSource = langDriver.createSqlSource(configuration, context, parameterTypeClass);
createSqlSource
就会调用
1 2 3 4 5 @Override public SqlSource createSqlSource (Configuration configuration, XNode script, Class<?> parameterType) { XMLScriptBuilder builder = new XMLScriptBuilder (configuration, script, parameterType); return builder.parseScriptNode(); }
再往下的逻辑在 parseScriptNode
中,org.apache.ibatis.scripting.xmltags.XMLScriptBuilder#parseScriptNode
1 2 3 4 5 6 7 8 9 10 public SqlSource parseScriptNode () { MixedSqlNode rootSqlNode = parseDynamicTags(context); SqlSource sqlSource; if (isDynamic) { sqlSource = new DynamicSqlSource (configuration, rootSqlNode); } else { sqlSource = new RawSqlSource (configuration, rootSqlNode, parameterType); } return sqlSource; }
首先要解析dynamicTag
,调用了org.apache.ibatis.scripting.xmltags.XMLScriptBuilder#parseDynamicTags
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 protected MixedSqlNode parseDynamicTags (XNode node) { List<SqlNode> contents = new ArrayList <>(); NodeList children = node.getNode().getChildNodes(); for (int i = 0 ; i < children.getLength(); i++) { XNode child = node.newXNode(children.item(i)); if (child.getNode().getNodeType() == Node.CDATA_SECTION_NODE || child.getNode().getNodeType() == Node.TEXT_NODE) { String data = child.getStringBody("" ); TextSqlNode textSqlNode = new TextSqlNode (data); if (textSqlNode.isDynamic()) { contents.add(textSqlNode); isDynamic = true ; } else { contents.add(new StaticTextSqlNode (data)); } } else if (child.getNode().getNodeType() == Node.ELEMENT_NODE) { String nodeName = child.getNode().getNodeName(); NodeHandler handler = nodeHandlerMap.get(nodeName); if (handler == null ) { throw new BuilderException ("Unknown element <" + nodeName + "> in SQL statement." ); } handler.handleNode(child, contents); isDynamic = true ; } } return new MixedSqlNode (contents); }
判断是否是动态sql
,调用了org.apache.ibatis.scripting.xmltags.TextSqlNode#isDynamic
1 2 3 4 5 6 7 public boolean isDynamic () { DynamicCheckerTokenParser checker = new DynamicCheckerTokenParser (); GenericTokenParser parser = createParser(checker); parser.parse(text); return checker.isDynamic(); }
创建parser
的时候可以看到这个parser
是干了啥,其实就是找有没有${
, }
1 2 3 private GenericTokenParser createParser (TokenHandler handler) { return new GenericTokenParser ("${" , "}" , handler); }
如果是的话,就在上面把 isDynamic
设置为true
如果是true
的话就创建 DynamicSqlSource
1 sqlSource = new DynamicSqlSource (configuration, rootSqlNode);
如果不是的话就创建RawSqlSource
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 sqlSource = new RawSqlSource (configuration, rootSqlNode, parameterType); ```java 但是这不是一个真实可用的 `sqlSource` , 实际创建的时候会走到这 ```java public RawSqlSource (Configuration configuration, SqlNode rootSqlNode, Class<?> parameterType) { this (configuration, getSql(configuration, rootSqlNode), parameterType); } public RawSqlSource (Configuration configuration, String sql, Class<?> parameterType) { SqlSourceBuilder sqlSourceParser = new SqlSourceBuilder (configuration); Class<?> clazz = parameterType == null ? Object.class : parameterType; sqlSource = sqlSourceParser.parse(sql, clazz, new HashMap <>()); }
具体的sqlSource
是通过org.apache.ibatis.builder.SqlSourceBuilder#parse
创建的 具体的代码逻辑是
1 2 3 4 5 6 7 8 9 10 11 public SqlSource parse (String originalSql, Class<?> parameterType, Map<String, Object> additionalParameters) { ParameterMappingTokenHandler handler = new ParameterMappingTokenHandler (configuration, parameterType, additionalParameters); GenericTokenParser parser = new GenericTokenParser ("#{" , "}" , handler); String sql; if (configuration.isShrinkWhitespacesInSql()) { sql = parser.parse(removeExtraWhitespaces(originalSql)); } else { sql = parser.parse(originalSql); } return new StaticSqlSource (configuration, sql, handler.getParameterMappings()); }
这里创建的其实是StaticSqlSource
,多带一句前面的parser
是将原来这样select * from student where id = #{id}
的 sql
解析成了select * from student where id = ?
然后创建了StaticSqlSource
1 2 3 4 5 public StaticSqlSource (Configuration configuration, String sql, List<ParameterMapping> parameterMappings) { this .sql = sql; this .parameterMappings = parameterMappings; this .configuration = configuration; }
为什么前面要讲这么多好像没什么关系的代码呢,其实在最开始我们执行sql
的代码中
1 2 3 4 5 6 @Override public <E> List<E> query (MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException { BoundSql boundSql = ms.getBoundSql(parameterObject); CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql); return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); }
这里获取了BoundSql
,而BoundSql
是怎么来的呢,首先调用了org.apache.ibatis.mapping.MappedStatement#getBoundSql
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 public BoundSql getBoundSql (Object parameterObject) { BoundSql boundSql = sqlSource.getBoundSql(parameterObject); List<ParameterMapping> parameterMappings = boundSql.getParameterMappings(); if (parameterMappings == null || parameterMappings.isEmpty()) { boundSql = new BoundSql (configuration, boundSql.getSql(), parameterMap.getParameterMappings(), parameterObject); } for (ParameterMapping pm : boundSql.getParameterMappings()) { String rmId = pm.getResultMapId(); if (rmId != null ) { ResultMap rm = configuration.getResultMap(rmId); if (rm != null ) { hasNestedResultMaps |= rm.hasNestedResultMaps(); } } } return boundSql; }
而我们从上面的解析中可以看到这里的sqlSource
是一层RawSqlSource
, 它的getBoundSql
又是调用内部的sqlSource
的方法
1 2 3 4 @Override public BoundSql getBoundSql (Object parameterObject) { return sqlSource.getBoundSql(parameterObject); }
内部的sqlSource
就是StaticSqlSource
,
1 2 3 4 @Override public BoundSql getBoundSql (Object parameterObject) { return new BoundSql (configuration, sql, parameterMappings, parameterObject); }
这个BoundSql
的内容也比较简单
1 2 3 4 5 6 7 public BoundSql (Configuration configuration, String sql, List<ParameterMapping> parameterMappings, Object parameterObject) { this .sql = sql; this .parameterMappings = parameterMappings; this .parameterObject = parameterObject; this .additionalParameters = new HashMap <>(); this .metaParameters = configuration.newMetaObject(additionalParameters); }
而上次在这边org.apache.ibatis.executor.SimpleExecutor#doQuery
的时候落了个东西,就是StatementHandler
的逻辑
1 2 3 4 5 6 7 8 9 10 11 12 @Override 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); } }
它是通过statementType来区分应该使用哪个statementHandler,我们这使用的就是PreparedStatementHandler
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 public RoutingStatementHandler (Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) { 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()); } }
所以上次有个细节可以补充,这边的doQuery里面的handler.query 应该是调用了PreparedStatementHandler 的query方法
1 2 3 4 5 6 7 8 9 10 11 12 @Override 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); } }
因为上面prepareStatement中getConnection拿到connection是com.mysql.cj.jdbc.ConnectionImpl#ConnectionImpl(com.mysql.cj.conf.HostInfo)
1 2 3 4 5 6 @Override public <E> List<E> query (Statement statement, ResultHandler resultHandler) throws SQLException { PreparedStatement ps = (PreparedStatement) statement; ps.execute(); return resultSetHandler.handleResultSets(ps); }
那又为什么是这个呢,可以在网上找,我们在mybatis-config.xml里配置的
1 <transactionManager type ="JDBC" />
因此在parseConfiguration中配置environment时
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 private void parseConfiguration (XNode root) { try { propertiesElement(root.evalNode("properties" )); Properties settings = settingsAsProperties(root.evalNode("settings" )); loadCustomVfs(settings); loadCustomLogImpl(settings); typeAliasesElement(root.evalNode("typeAliases" )); pluginElement(root.evalNode("plugins" )); objectFactoryElement(root.evalNode("objectFactory" )); objectWrapperFactoryElement(root.evalNode("objectWrapperFactory" )); reflectorFactoryElement(root.evalNode("reflectorFactory" )); settingsElement(settings); environmentsElement(root.evalNode("environments" )); databaseIdProviderElement(root.evalNode("databaseIdProvider" )); typeHandlerElement(root.evalNode("typeHandlers" )); mapperElement(root.evalNode("mappers" )); } catch (Exception e) { throw new BuilderException ("Error parsing SQL Mapper Configuration. Cause: " + e, e); } }
调用的这个方法通过获取xml中的transactionManager 配置的类型,也就是JDBC
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 private void environmentsElement (XNode context) throws Exception { if (context != null ) { if (environment == null ) { environment = context.getStringAttribute("default" ); } for (XNode child : context.getChildren()) { String id = child.getStringAttribute("id" ); if (isSpecifiedEnvironment(id)) { TransactionFactory txFactory = transactionManagerElement(child.evalNode("transactionManager" )); DataSourceFactory dsFactory = dataSourceElement(child.evalNode("dataSource" )); DataSource dataSource = dsFactory.getDataSource(); Environment.Builder environmentBuilder = new Environment .Builder(id) .transactionFactory(txFactory) .dataSource(dataSource); configuration.setEnvironment(environmentBuilder.build()); break ; } } } }
是通过以下方法获取的,
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 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 private TransactionFactory transactionManagerElement (XNode context) throws Exception { if (context != null ) { String type = context.getStringAttribute("type" ); Properties props = context.getChildrenAsProperties(); TransactionFactory factory = (TransactionFactory) resolveClass(type).getDeclaredConstructor().newInstance(); factory.setProperties(props); return factory; } throw new BuilderException ("Environment declaration requires a TransactionFactory." ); } protected <T> Class<? extends T > resolveClass(String alias) { if (alias == null ) { return null ; } try { return resolveAlias(alias); } catch (Exception e) { throw new BuilderException ("Error resolving class. Cause: " + e, e); } } protected <T> Class<? extends T > resolveAlias(String alias) { return typeAliasRegistry.resolveAlias(alias); } public <T> Class<T> resolveAlias (String string) { try { if (string == null ) { return null ; } String key = string.toLowerCase(Locale.ENGLISH); Class<T> value; if (typeAliases.containsKey(key)) { value = (Class<T>) typeAliases.get(key); } else { value = (Class<T>) Resources.classForName(string); } return value; } catch (ClassNotFoundException e) { throw new TypeException ("Could not resolve type alias '" + string + "'. Cause: " + e, e); } }
而通过JDBC获取得是啥的,就是在Configuration的构造方法里写了的JdbcTransactionFactory
1 2 public Configuration () { typeAliasRegistry.registerAlias("JDBC" , JdbcTransactionFactory.class);
所以我们在这
1 2 3 4 5 private SqlSession openSessionFromDataSource (ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) { Transaction tx = null ; try { final Environment environment = configuration.getEnvironment(); final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
获得到的TransactionFactory 就是 JdbcTransactionFactory ,而后
1 2 3 4 5 6 7 8 9 tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit); ```java 创建的transaction就是JdbcTransaction ```java @Override public Transaction newTransaction (DataSource ds, TransactionIsolationLevel level, boolean autoCommit) { return new JdbcTransaction (ds, level, autoCommit, skipSetAutoCommitOnClose); }
然后我们再会上去看代码getConnection ,
1 2 3 4 5 6 7 8 9 protected Connection getConnection (Log statementLog) throws SQLException { Connection connection = transaction.getConnection(); if (statementLog.isDebugEnabled()) { return ConnectionLogger.newInstance(connection, statementLog, queryStack); } else { return connection; } }
即调用了
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 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 @Override public Connection getConnection () throws SQLException { if (connection == null ) { openConnection(); } return connection; } protected void openConnection () throws SQLException { if (log.isDebugEnabled()) { log.debug("Opening JDBC Connection" ); } connection = dataSource.getConnection(); if (level != null ) { connection.setTransactionIsolation(level.getLevel()); } setDesiredAutoCommit(autoCommit); } @Override public Connection getConnection () throws SQLException { return popConnection(dataSource.getUsername(), dataSource.getPassword()).getProxyConnection(); } private PooledConnection popConnection (String username, String password) throws SQLException { boolean countedWait = false ; PooledConnection conn = null ; long t = System.currentTimeMillis(); int localBadConnectionCount = 0 ; while (conn == null ) { lock.lock(); try { if (!state.idleConnections.isEmpty()) { conn = state.idleConnections.remove(0 ); if (log.isDebugEnabled()) { log.debug("Checked out connection " + conn.getRealHashCode() + " from pool." ); } } else { if (state.activeConnections.size() < poolMaximumActiveConnections) { conn = new PooledConnection (dataSource.getConnection(), this ); if (log.isDebugEnabled()) { log.debug("Created connection " + conn.getRealHashCode() + "." ); } } else { PooledConnection oldestActiveConnection = state.activeConnections.get(0 ); long longestCheckoutTime = oldestActiveConnection.getCheckoutTime(); if (longestCheckoutTime > poolMaximumCheckoutTime) { state.claimedOverdueConnectionCount++; state.accumulatedCheckoutTimeOfOverdueConnections += longestCheckoutTime; state.accumulatedCheckoutTime += longestCheckoutTime; state.activeConnections.remove(oldestActiveConnection); if (!oldestActiveConnection.getRealConnection().getAutoCommit()) { try { oldestActiveConnection.getRealConnection().rollback(); } catch (SQLException e) { log.debug("Bad connection. Could not roll back" ); } } conn = new PooledConnection (oldestActiveConnection.getRealConnection(), this ); conn.setCreatedTimestamp(oldestActiveConnection.getCreatedTimestamp()); conn.setLastUsedTimestamp(oldestActiveConnection.getLastUsedTimestamp()); oldestActiveConnection.invalidate(); if (log.isDebugEnabled()) { log.debug("Claimed overdue connection " + conn.getRealHashCode() + "." ); } } else { try { if (!countedWait) { state.hadToWaitCount++; countedWait = true ; } if (log.isDebugEnabled()) { log.debug("Waiting as long as " + poolTimeToWait + " milliseconds for connection." ); } long wt = System.currentTimeMillis(); condition.await(poolTimeToWait, TimeUnit.MILLISECONDS); state.accumulatedWaitTime += System.currentTimeMillis() - wt; } catch (InterruptedException e) { Thread.currentThread().interrupt(); break ; } } } } if (conn != null ) { if (conn.isValid()) { if (!conn.getRealConnection().getAutoCommit()) { conn.getRealConnection().rollback(); } conn.setConnectionTypeCode(assembleConnectionTypeCode(dataSource.getUrl(), username, password)); conn.setCheckoutTimestamp(System.currentTimeMillis()); conn.setLastUsedTimestamp(System.currentTimeMillis()); state.activeConnections.add(conn); state.requestCount++; state.accumulatedRequestTime += System.currentTimeMillis() - t; } else { if (log.isDebugEnabled()) { log.debug("A bad connection (" + conn.getRealHashCode() + ") was returned from the pool, getting another connection." ); } state.badConnectionCount++; localBadConnectionCount++; conn = null ; if (localBadConnectionCount > (poolMaximumIdleConnections + poolMaximumLocalBadConnectionTolerance)) { if (log.isDebugEnabled()) { log.debug("PooledDataSource: Could not get a good connection to the database." ); } throw new SQLException ("PooledDataSource: Could not get a good connection to the database." ); } } } } finally { lock.unlock(); } } if (conn == null ) { if (log.isDebugEnabled()) { log.debug("PooledDataSource: Unknown severe error condition. The connection pool returned a null connection." ); } throw new SQLException ("PooledDataSource: Unknown severe error condition. The connection pool returned a null connection." ); } return conn; }
其实就是调用的
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 @Override public Connection getConnection () throws SQLException { return doGetConnection(username, password); } ```java 然后就是 ```java private Connection doGetConnection (String username, String password) throws SQLException { Properties props = new Properties (); if (driverProperties != null ) { props.putAll(driverProperties); } if (username != null ) { props.setProperty("user" , username); } if (password != null ) { props.setProperty("password" , password); } return doGetConnection(props); }
继续这个逻辑
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 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 private Connection doGetConnection (Properties properties) throws SQLException { initializeDriver(); Connection connection = DriverManager.getConnection(url, properties); configureConnection(connection); return connection; } @CallerSensitive public static Connection getConnection (String url, java.util.Properties info) throws SQLException { return (getConnection(url, info, Reflection.getCallerClass())); } private static Connection getConnection ( String url, java.util.Properties info, Class<?> caller) throws SQLException { ClassLoader callerCL = caller != null ? caller.getClassLoader() : null ; synchronized (DriverManager.class) { if (callerCL == null ) { callerCL = Thread.currentThread().getContextClassLoader(); } } if (url == null ) { throw new SQLException ("The url cannot be null" , "08001" ); } println("DriverManager.getConnection(\"" + url + "\")" ); SQLException reason = null ; for (DriverInfo aDriver : registeredDrivers) { if (isDriverAllowed(aDriver.driver, callerCL)) { try { println(" trying " + aDriver.driver.getClass().getName()); Connection con = aDriver.driver.connect(url, info); if (con != null ) { println("getConnection returning " + aDriver.driver.getClass().getName()); return (con); } } catch (SQLException ex) { if (reason == null ) { reason = ex; } } } else { println(" skipping: " + aDriver.getClass().getName()); } } if (reason != null ) { println("getConnection failed: " + reason); throw reason; } println("getConnection: no suitable driver found for " + url); throw new SQLException ("No suitable driver found for " + url, "08001" ); }
上面的driver就是driver[className=com.mysql.cj.jdbc.Driver@64030b91]
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 32 33 public Connection connect (String url, Properties info) throws SQLException { try { try { if (!ConnectionUrl.acceptsUrl(url)) { return null ; } else { ConnectionUrl conStr = ConnectionUrl.getConnectionUrlInstance(url, info); switch (conStr.getType()) { case SINGLE_CONNECTION: return ConnectionImpl.getInstance(conStr.getMainHost()); case FAILOVER_CONNECTION: case FAILOVER_DNS_SRV_CONNECTION: return FailoverConnectionProxy.createProxyInstance(conStr); case LOADBALANCE_CONNECTION: case LOADBALANCE_DNS_SRV_CONNECTION: return LoadBalancedConnectionProxy.createProxyInstance(conStr); case REPLICATION_CONNECTION: case REPLICATION_DNS_SRV_CONNECTION: return ReplicationConnectionProxy.createProxyInstance(conStr); default : return null ; } } } catch (UnsupportedConnectionStringException var5) { return null ; } catch (CJException var6) { throw (UnableToConnectException)ExceptionFactory.createException(UnableToConnectException.class, Messages.getString("NonRegisteringDriver.17" , new Object []{var6.toString()}), var6); } } catch (CJException var7) { throw SQLExceptionsMapping.translateException(var7); } }
这是个 SINGLE_CONNECTION ,所以调用的就是 return ConnectionImpl.getInstance(conStr.getMainHost()); 然后在这里设置了代理类
1 2 3 4 5 6 7 8 9 public PooledConnection (Connection connection, PooledDataSource dataSource) { this .hashCode = connection.hashCode(); this .realConnection = connection; this .dataSource = dataSource; this .createdTimestamp = System.currentTimeMillis(); this .lastUsedTimestamp = System.currentTimeMillis(); this .valid = true ; this .proxyConnection = (Connection) Proxy.newProxyInstance(Connection.class.getClassLoader(), IFACES, this ); }
结合这个
1 2 3 4 @Override public Connection getConnection () throws SQLException { return popConnection(dataSource.getUsername(), dataSource.getPassword()).getProxyConnection(); }
所以最终的connection就是com.mysql.cj.jdbc.ConnectionImpl@358ab600