![MyBatis 3源码深度解析](https://wfqqreader-1252317822.image.myqcloud.com/cover/388/27563388/b_27563388.jpg)
3.1 使用SQL类生成语句
使用JDBC API开发过项目的读者应该知道,当我们需要使用Statement对象执行SQL时,SQL语句会嵌入Java代码中。SQL语句比较复杂时,我们可能会在代码中对SQL语句进行拼接,查询条件不固定时,还需要根据不同条件拼接不同的SQL语句,拼接语句时不要忘记添加必要的空格,还要注意去掉列表最后一个列名的逗号。这个过程对于开发人员来说简直就是一场噩梦,而且代码可维护性级低,例如:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P69_7116.jpg?sign=1738847274-x6i8qgQX0UzoZEkoq6BfEanvHMerzSr7-0-6366f9ea2521cd8c26553424066f56ad)
为了解决这个问题,MyBatis中提供了一个SQL工具类。使用这个工具类,我们可以很方便地在Java代码中动态构建SQL语句。上面的语句如果使用SQL工具类来构建,就会简单很多。下面是使用MyBatis中的SQL工具类动态构建SQL语句的案例:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P70_7119.jpg?sign=1738847274-1W60GeX5tlQcc4GjZwnGF7j0HtKc7lUh-0-bfaee63d80abbe8ce0c7baba79510cc7)
如上面的代码所示,创建了一个匿名的SQL类的子类,在匿名子类的初始化代码块中,调用SELECT()、FROM()等方法构建SQL语句,这种方式能够很好地避免字符串拼接过程中缺少空格或者偶然间重复出现的AND关键字导致的SQL语句不正确。
除了SELECT语句外,SQL工具类也可以用作构建UPDATE、INSERT等语句。下面是SQL工具类的一些使用案例:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P70_7121.jpg?sign=1738847274-H6oenBdzNNg4yW3d7S3rZBINz7feUH8W-0-ff6316efc99a8986a8d0a74482e6ce90)
使用SQL工具类的另一个好处是可以很方便地在Java代码中根据条件动态地拼接SQL语句,例如:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P71_7123.jpg?sign=1738847274-4EGclTwC9NvCB4eBXv3RrG6IHUTCyZSU-0-2568906adbfc484459890972c03e8fcd)
关于SQL工具类的完整使用案例可参考随书源码mybatis-chapter03项目的com.blog4java.mybatis.SQLExample案例。
SQL工具类中提供的所有方法及作用可参考表3-1中的内容。
表3-1 SQL工具类的方法及作用
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-T71_7124.jpg?sign=1738847274-Zor61t3wuDoer0g0D7kAEmaIUg7UpJ4m-0-05655496028249aa43f9808e24fbdd70)
(续表)
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-T72_7125.jpg?sign=1738847274-7xDr83ch746LyWfzbrK1hq6gz8P0ouDc-0-2b98bb93af7ff0ec09da764bd7d3f2b3)
在学习完SQL工具类的使用后,接下来我们简单地了解一下SQL工具类的实现源码。SQL继承至AbstractSQL类,只重写了该类的getSelf()方法,代码如下:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P72_7127.jpg?sign=1738847274-WHGG56E0XiUUvpwnTOXSqDUCdVMDST0M-0-6f51dea2b39b98266d889964309c0bf8)
所有的功能由AbstractSQL类完成,AbstractSQL类中维护了一个SQLStatement内部类的实例和一系列前面提到过的构造SQL语句的方法,例如SELECT()、UPDATE()等方法。AbstractSQL类的部分代码如下:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P73_4216.jpg?sign=1738847274-QjxHijBfiXFjHjOweRvwevQD4S2vGfyk-0-e32c571046164ddc3c80c0437adf239e)
SQLStatement内部类用于描述一个SQL语句,该类中通过StatementType确定SQL语句的类型。SQLStatement类中还维护了一系列的ArrayList属性,当调用SELECT()、UPDATE()等方法时,这些方法的参数内容会记录在这些ArrayList对象中,SQLStatement类中的属性如下:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P73_7129.jpg?sign=1738847274-MqYteVXqyelInswsxJCjdOLm1d8KJCGg-0-9adfd8796b56d9743bdf897c8f29ccb8)
AbstrastSQL类重写了toString()方法,该方法中会调用SQLStatement对象的sql()方法生成SQL字符串,代码如下:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P74_7131.jpg?sign=1738847274-snIzkl8b5pYA3dG8bbdlmJRyNzkfDTW2-0-e7550257e47eb6cdb681a3d17540ddab)
SQLStatement对象的sql()方法实现代码如下:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P74_7134.jpg?sign=1738847274-EoiK0uMEH5UxO64C1Zz2UchlWBYNIWuW-0-73a93367a19b96a8c942a81a3bcedb36)
如上面的代码所示,该方法中会判断SQL语句的类型,以UPDATE语句为例,会调用SQLStatement对象的updateSql()方法生成UPDATE语句。updateSql()方法代码如下:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P74_7136.jpg?sign=1738847274-xLgu2BuMJp0878pWsA5XM84sZuA1sBEM-0-7e808c9a9605561d92f703dde56eba41)
如上面的代码所示,updateSql()方法中,最终会调用sqlCalause()方法完成SQL语句的拼接。sqlCalause()方法实现代码如下:
![](https://epubservercos.yuewen.com/898509/15825993104147706/epubprivate/OEBPS/Images/Figure-P74_7138.jpg?sign=1738847274-ZbYPkkOsjELEs23hUqDFj8bKpOZbCEEI-0-a8254dec99481add838ab6a6c0c352d4)
这里对SQL工具类的实现做了简单的介绍,完整代码可参考AbstractSQL类的源码。