Mybatis自定義SQL的關(guān)系映射、分頁(yè)、排序功能的實(shí)現(xiàn)
目的: 記錄數(shù)據(jù)庫(kù)表與實(shí)體對(duì)象之間不同的映射關(guān)系如何用mybatis的自定義sql和結(jié)果返回集處理。
1、三種對(duì)象映射關(guān)系
1.1 一對(duì)一
一個(gè)人對(duì)應(yīng)一個(gè)身份證,一位同學(xué)對(duì)應(yīng)一個(gè)班級(jí),每個(gè)房間都有自己的房間號(hào),當(dāng)一個(gè)事物它對(duì)應(yīng)另一個(gè)事物是唯一的,那么它們之間的關(guān)系就是一對(duì)一的。
這里我演示的案例是,一個(gè)學(xué)生有著一位老師
老師基礎(chǔ)信息:
學(xué)生詳細(xì)信息:
如果說(shuō),我們需要將兩個(gè)表一起查出來(lái),我們可以這么做:
問(wèn)題: 如果對(duì)象的列重復(fù)了,必須要使用到別名
1、先定義實(shí)體結(jié)構(gòu),也就是我們返結(jié)果的實(shí)體類
public class Student { @TableId private int id; private String name; private int tid; @TableField(exist = false) private Teacher teacher; }
Teacher:
public class Teacher { @TableId private int id; private String name; }
2、 編寫(xiě)xml文件
這里有兩種方式,使用association時(shí)的關(guān)鍵在于告訴mybatis如何加載關(guān)聯(lián)(assocition)。
- 嵌套查詢:通過(guò)執(zhí)行另外一個(gè) SQL 映射語(yǔ)句來(lái)返回預(yù)期的復(fù)雜類型。
- 嵌套結(jié)果:使用嵌套結(jié)果映射來(lái)處理重復(fù)的聯(lián)合結(jié)果的子集。
第一種: 使用嵌套查詢,也就是使用另一個(gè)sql
// teacherMapper.xml <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.lll.mybatisplusdemo.mapper.TeacherMapper"> <select id="getTeacher" parameterType="int" resultType="teacher"> select * from teacher where id = #{id}; </select> </mapper> // studentMapper.xml <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <select id="getStudent2" parameterType="int" resultMap="getStudentMap2"> select * from student where id =#{id}; </select> <resultMap id="getStudentMap2" type="student"> <id column="id" property="id"></id> <result column="name" property="name"></result> <result column="tid" property="tid"></result> <association property="teacher" javaType="Teacher" column="tid" select="com.lll.mybatisplusdemo.mapper.TeacherMapper.getTeacher"> <id column="id" property="id"></id> <result column="name" property="name"></result> </association> </resultMap> </mapper>
嵌套查詢的方式很簡(jiǎn)單,但是對(duì)于大型數(shù)據(jù)集合和列表將不會(huì)表現(xiàn)很好。問(wèn)題就是我們熟知的
“N+1 查詢問(wèn)題”。概括地講, N+1 查詢問(wèn)題可以是這樣引起的:
- 你執(zhí)行了一個(gè)單獨(dú)的 SQL 語(yǔ)句來(lái)獲取結(jié)果列表(就是“+1”)。
- 對(duì)返回的每條記錄,你執(zhí)行了一個(gè)查詢語(yǔ)句來(lái)為每個(gè)加載細(xì)節(jié)(就是“N”)。
第二種: 使用嵌套結(jié)果來(lái)映射聯(lián)合查詢來(lái)的數(shù)據(jù)
<?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.lll.mybatisplusdemo.mapper.StudentMapper"> <select id="getStudent" parameterType="int" resultMap="getStudentMap"> SELECT a.*,b.id as cid,b.name as cname FROM `student` as a,teacher as b WHERE a.id =#{id} and a.tid = b.id; </select> <resultMap id="getStudentMap" type="student"> <id column="id" property="id"></id> <result column="name" property="name"></result> <result column="tid" property="tid"></result> <association property="teacher" javaType="Teacher"> <id column="cid" property="id"></id> <result column="cname" property="name"></result> </association> </resultMap> </mapper>
我們?cè)谙鄳?yīng)的mapper中添加方法接口便可以使用了。
1.2 一對(duì)多
案例:一個(gè)老師有多個(gè)學(xué)生
1、實(shí)體類
public class Teacher { @TableId private int id; private String name; @TableField(exist = false) private List<Student> students; }
2、編寫(xiě)xml
同樣還是,我們先來(lái)個(gè)嵌套結(jié)果映射
嵌套結(jié)果:
// teacherMapper.xml <select id="getStudent" parameterType="int" resultMap="getStudentMap"> SELECT a.*,b.id as cid,b.name as cname,b.tid from teacher as a , student as b where b.tid = a.id and a.id =#{id}; </select> <resultMap id="getStudentMap" type="teacher"> <id column="id" property="id"></id> <result column="name" property="name"></result> <collection property="students" ofType="Student"> <id column="cid" property="id"></id> <result column="cname" property="name"></result> <result column="tid" property="tid" ></result> </collection> </resultMap>
嵌套查詢:
// teacherMapper.xml <select id="getStudent2" parameterType="int" resultMap="getStudentMap2"> select * from teacher as a where a.id = #{id} </select> <resultMap id="getStudentMap2" type="teacher"> <id column="id" property="id"></id> <result column="name" property="name"></result> <collection property="students" column="id" ofType="Student" select="com.lll.mybatisplusdemo.mapper.StudentMapper.getStudentByTid"> </collection> </resultMap> // studentMapper.xml <select id="getStudentByTid" parameterType="int" resultType="student"> select * from student as a where a.tid = #{id} </select>
1.3 多對(duì)多
學(xué)生與課程是多對(duì)多的關(guān)系,與上面的一對(duì)多的操作方式是類似的
2、自定義sql如何做分頁(yè)
mapper定義方法,方法傳入page參數(shù)
public interface UserMapper{ IPage<User> selectPageVo(Page<User> page); }
userMapper.xml文件編寫(xiě)一個(gè)普通的返回結(jié)果是list的方法,mybatis會(huì)自動(dòng)幫你做分頁(yè)
<select id="selectPage" resultType="com.baomidou.cloud.entity.UserVo"> SELECT * FROM user </select>
3、自定義sql如何做排序
結(jié)論:使用order by,記住要使用'$',不能使用'#'
<select id="selectPage" resultType="com.baomidou.cloud.entity.UserVo"> SELECT * FROM user order by ${sortColumn} ${sortOrder} </select>
4、自定義sql中的#{}和${}的區(qū)別
1、傳入的參數(shù)在SQL中顯示不同
#傳入的參數(shù)在SQL中顯示為字符串(當(dāng)成一個(gè)字符串),會(huì)對(duì)自動(dòng)傳入的數(shù)據(jù)加一個(gè)雙引號(hào)。
例:使用以下SQL
select id,name,age from student where id =#{id}
當(dāng)我們傳遞的參數(shù)id為 “1” 時(shí),上述 sql 的解析為:
select id,name,age from student where id ="1"
$傳入的參數(shù)在SqL中直接顯示為傳入的值
例:使用以下SQL
select id,name,age from student where id =${id}
當(dāng)我們傳遞的參數(shù)id為 “1” 時(shí),上述 sql 的解析為:
select id,name,age from student where id =1
2、#可以防止SQL注入的風(fēng)險(xiǎn)(語(yǔ)句的拼接);但$無(wú)法防止Sql注入。
3、$方式一般用于傳入數(shù)據(jù)庫(kù)對(duì)象,例如傳入表名。
4、大多數(shù)情況下還是經(jīng)常使用#,一般能用#的就別用$
;但有些情況下必須使用$
,例:MyBatis排序時(shí)使用order by 動(dòng)態(tài)參數(shù)時(shí)需要注意,用$而不是#。
到此這篇關(guān)于Mybatis自定義SQL的關(guān)系映射、分頁(yè)、排序的文章就介紹到這了,更多相關(guān)Mybatis自定義SQL的關(guān)系映射內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java web中使用cookie記住用戶的賬號(hào)和密碼
這篇文章主要介紹了java web中使用cookie記住用戶的賬號(hào)和密碼的相關(guān)資料,需要的朋友可以參考下2017-01-01Java中List、Set、Map的區(qū)別和實(shí)現(xiàn)方式示例代碼
這篇文章主要介紹了Java中List、Set、Map的區(qū)別和實(shí)現(xiàn)方式示例代碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06jmeter添加自定函數(shù)的實(shí)例(jmeter5.3+IntelliJ IDEA)
這篇文章主要介紹了jmeter添加自定函數(shù)的實(shí)例(jmeter5.3+IntelliJ IDEA),本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11使用jaxp進(jìn)行dom解析_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
這篇文章主要介紹了使用jaxp進(jìn)行dom解析的相關(guān)資料,需要的朋友可以參考下2017-08-08SpringMVC空指針異常NullPointerException解決及原理解析
這篇文章主要介紹了SpringMVC空指針異常NullPointerException解決及原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08劍指Offer之Java算法習(xí)題精講數(shù)組與字符串
跟著思路走,之后從簡(jiǎn)單題入手,反復(fù)去看,做過(guò)之后可能會(huì)忘記,之后再做一次,記不住就反復(fù)做,反復(fù)尋求思路和規(guī)律,慢慢積累就會(huì)發(fā)現(xiàn)質(zhì)的變化2022-03-03JAVA實(shí)現(xiàn)簡(jiǎn)單停車場(chǎng)系統(tǒng)代碼
JAVA項(xiàng)目中正號(hào)需要一個(gè)停車收費(fèi)系統(tǒng),就整理出來(lái)java實(shí)現(xiàn)的一個(gè)簡(jiǎn)單的停車收費(fèi)系統(tǒng)給大家分享一下,希望對(duì)大家有所幫助2017-04-04