图比较大:如果看着比较糊的话,可以下载高清图:https://download.csdn.net/download/langwuzhe/87376216
第一步:创建 StatementHandler、ParameterHandler、ResultSetHandler-----------(三剑客的新生)
- 创建 StatementHandler 对象
- 具体创建的实现类是
PreparedStatementHandler
,而且被RoutingStatementHandler
包裹,便于识别具体的StatementHandler
对象。
- 具体创建的实现类是
- 在创建 StatementHandler 对象时,会同时super()父类
BaseStatementHandler
的构造器。BaseStatementHandler
构造器会把 ParameterHandler 、ResultSetHandler 都创建出来- ParameterHandler 、ResultSetHandler 这两个Handler 是在StatementHandler的实现类中创建出来的,所以当要调用 这两个Handler 的方法时需要去 StatementHandler 的实现中寻找这两个Hander 的引用。所以就会看见上图中 设置参数 和 整理数据库返回数据时依然会看见需要经过 StatementHandler的实现类PrepareStatementHandler
第二步:执行 connection.prepareStatement,返回 statement 对象---------(StatementHandler)
- 经过一系列流转,具体是在 StatementHandler 的实现类中执行的
connection.prepareStatement
- 在
connection.prepareStatement(sql)
时,由于connection 对象被种了动态代理的蛊。所以执行connection.prepareStatement(sql)
时,会先进入 connection的代理对象 ConnectionLogger中执行invoke方法 打印sql日志
第三步:往sql中设置参数。具体值 替换 ? 问号 -------------- (ParameterHandler)
- 把第二步的
statement
对象传出来用来往sql设置参数 - 使用具体参数 替换 ? 问号时,是在ParameterHandler的实现类
DefaultParameterHandler
进行的
第四步:执行 statement.execute() ------------------------(StatementHandler)
- 执行
execute()
方法是在 StatementHandler 的实现类 PreparedStatementHandler 中执行的。 - 由于 PreparedStatement 对象被种了动态代理的蛊,所以执行 他的
execute()
方法,会先进入到PreparedStatement
的代理的对象PreparedStatementLogger
中打印入参的日志。
第五步:数据库返回数据结果映射。---------------------------(ResultSetHandler)
-
结果映射都是在 ResultSetHandler 的实现类 DefaultResultSetHandler 中进行的。
-
从 statement中取出 ResultSet 对象会把他放到
ResultSetWrapper
中,ResultSetWrapper
是对ResultSet 的补充增强 -
在映射的过程中,还会创建 ResultHandler对象,他的实现类是
DefaultResultHandler
。这个只是方法返回时携带数据用的,无需过多关注,但是要区分ResultSetHandler 和 ResultHandler 长的很像,容易看花眼。
注:
第一、二、三 步 都是在 Executor 中发生的。第四、五步 是在 StatementHandler 中发生的。