您好,登錄后才能下訂單哦!
這篇文章主要為大家展示了“MyBatis中XML基本用法之多表查詢功能的示例分析”,內(nèi)容簡(jiǎn)而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“MyBatis中XML基本用法之多表查詢功能的示例分析”這篇文章吧。
在之前,我們示例的2個(gè)查詢都是單表查詢,但實(shí)際的業(yè)務(wù)場(chǎng)景肯定是需要多表查詢的,比如現(xiàn)在有個(gè)需求:
查詢某個(gè)用戶擁有的所有角色。這個(gè)需求要涉及到sys_user,sys_user_role,sys_role三張表,如何實(shí)現(xiàn)呢?
首先,在SysUserMapper接口中定義如下方法。
/** * 根據(jù)用戶id獲取角色信息 * * @param userId * @return */ List<SysRole> selectRolesByUserId(Long userId);
然后打開對(duì)應(yīng)的SysUserMapper.xml文件,添加如下select語句:
<select id="selectRolesByUserId" resultType="com.zwwhnly.mybatisaction.model.SysRole"> SELECT r.id, r.role_name roleName, r.enabled, r.create_by createBy, r.create_time createTime FROM sys_user u INNER JOIN sys_user_role ur ON u.id = ur.user_id INNER JOIN sys_role r ON ur.role_id = r.id WHERE u.id = #{userId} </select>
細(xì)心的讀者可能會(huì)發(fā)現(xiàn),我們雖然使用到了多表查詢,但是resultType設(shè)置的仍然是單表,即只包含角色表的信息。
如果我希望這個(gè)查詢語句同時(shí)返回SysUser表的user_name字段呢,該如何設(shè)置resultType?
方法1:直接在SysRole實(shí)體類中添加userName字段。
private String userName; public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; }
此時(shí)resultType不用修改。
方法2:新建擴(kuò)展類,在擴(kuò)展類中添加userName字段。
package com.zwwhnly.mybatisaction.model; public class SysRoleExtend extends SysRole { private String userName; public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } }
此時(shí)需要將resultType修改為:com.zwwhnly.mybatisaction.model.SysRoleExtend
。
這種方式比較適合需要少量額外字段的場(chǎng)景。如果需要其他表的大量字段,可以使用方式3或者方式4,個(gè)人推薦使用方式4。
方法3:在SysRole實(shí)體類中添加SysUser類型的字段。
private SysUser sysUser; public SysUser getSysUser() { return sysUser; } public void setSysUser(SysUser sysUser) { this.sysUser = sysUser; }
此時(shí)resultType不用修改。
方法4(推薦使用):新建擴(kuò)展類,在擴(kuò)展類中添加SysUser類型的字段。
書中推薦的是方式3,方式4是我個(gè)人認(rèn)為更好的方式,因?yàn)閷?shí)體類一般由工具自動(dòng)生成,增加了字段后,后續(xù)容易忘記導(dǎo)致被覆蓋掉。
package com.zwwhnly.mybatisaction.model; public class SysRoleExtend extends SysRole { private SysUser sysUser; public SysUser getSysUser() { return sysUser; } public void setSysUser(SysUser sysUser) { this.sysUser = sysUser; } }
此時(shí)需要將resultType修改為:com.zwwhnly.mybatisaction.model.SysRoleExtend
。
此時(shí)xml中的查詢語句如下。
<select id="selectRolesByUserId" resultType="com.zwwhnly.mybatisaction.model.SysRoleExtend"> SELECT r.id, r.role_name roleName, r.enabled, r.create_by createBy, r.create_time createTime, u.user_name "sysUser.userName", u.user_email "sysUser.userEmail" FROM sys_user u INNER JOIN sys_user_role ur ON u.id = ur.user_id INNER JOIN sys_role r ON ur.role_id = r.id WHERE u.id = #{userId} </select>
在SysUserMapperTest中添加測(cè)試代碼如下。
@Test public void testSelectRolesByUserId() { SqlSession sqlSession = getSqlSession(); try { SysUserMapper sysUserMapper = sqlSession.getMapper(SysUserMapper.class); List<SysRole> sysRoleList = sysUserMapper.selectRolesByUserId(1L); Assert.assertNotNull(sysRoleList); Assert.assertTrue(sysRoleList.size() > 0); } finally { sqlSession.close(); } }
運(yùn)行該測(cè)試方法,輸入日志如下。
DEBUG [main] - ==> Preparing: SELECT r.id, r.role_name roleName, r.enabled, r.create_by createBy, r.create_time createTime, u.user_name "sysUser.userName", u.user_email "sysUser.userEmail" FROM sys_user u INNER JOIN sys_user_role ur ON u.id = ur.user_id INNER JOIN sys_role r ON ur.role_id = r.id WHERE u.id = ? DEBUG [main] - ==> Parameters: 1(Long) TRACE [main] - <== Columns: id, roleName, enabled, createBy, createTime, sysUser.userName, sysUser.userEmail TRACE [main] - <== Row: 1, 管理員, 1, 1, 2019-06-27 18:21:12.0, admin, admin@mybatis.tk TRACE [main] - <== Row: 2, 普通用戶, 1, 1, 2019-06-27 18:21:12.0, admin, admin@mybatis.tk DEBUG [main] - <== Total: 2
2.1 參數(shù)類型是基本類型
截止目前,我們定義的方法都只有1個(gè)參數(shù),要么是只有1個(gè)基本類型的參數(shù),比如selectById(Long id);。
要么是只有1個(gè)對(duì)象作為參數(shù),即將多個(gè)參數(shù)合并成了1個(gè)對(duì)象。
但有些場(chǎng)景下,比如只有2個(gè)參數(shù),沒有必要為這2個(gè)參數(shù)再新建一個(gè)對(duì)象,比如我們現(xiàn)在需要根據(jù)用戶的id和角色的狀態(tài)來獲取用戶的所有角色,那么該如何使用呢?
首先,在接口SysUserMapper中添加如下方法。
/** * 根據(jù)用戶id和角色的enabled狀態(tài)獲取用戶的角色 * * @param userId * @param enabled * @return */ List<SysRole> selectRolesByUserIdAndRoleEnabled(Long userId,Integer enabled);
然后,打開對(duì)應(yīng)的SysUserMapper.xml文件,添加如下代碼。
<select id="selectRolesByUserIdAndRoleEnabled" resultType="com.zwwhnly.mybatisaction.model.SysRole"> SELECT r.id, r.role_name roleName, r.enabled, r.create_by createBy, r.create_time createTime FROM sys_user u INNER JOIN sys_user_role ur ON u.id = ur.user_id INNER JOIN sys_role r ON ur.role_id = r.id WHERE u.id = #{userId} AND r.enabled = #{enabled} </select>
在SysUserMapperTest測(cè)試類中,添加如下測(cè)試方法。
@Test public void testselectRolesByUserIdAndRoleEnabled() { SqlSession sqlSession = getSqlSession(); try { SysUserMapper sysUserMapper = sqlSession.getMapper(SysUserMapper.class); List<SysRole> sysRoleList = sysUserMapper.selectRolesByUserIdAndRoleEnabled(1L, 1); Assert.assertNotNull(sysRoleList); Assert.assertTrue(sysRoleList.size() > 0); } finally { sqlSession.rollback(); sqlSession.close(); } }
運(yùn)行該測(cè)試方法,發(fā)現(xiàn)報(bào)如下錯(cuò)誤。
報(bào)錯(cuò)信息中說未找到參數(shù)userId,可用的參數(shù)是[0,1,param1,param2],也就是說我們將代碼修改為:
WHERE u.id = #{0} AND r.enabled = #{1}
或者修改為:
WHERE u.id = #{param1} AND r.enabled = #{param2}
這么使用是可以測(cè)試通過的,不過這樣使用,代碼閱讀起來不夠友好,因此并不推薦這么使用。
推薦在接口方法的參數(shù)前添加@Param注解,如下所示:
/** * 根據(jù)用戶id和角色的enabled狀態(tài)獲取用戶的角色 * * @param userId * @param enabled * @return */ List<SysRole> selectRolesByUserIdAndRoleEnabled(@Param("userId") Long userId, @Param("enabled") Integer enabled);
運(yùn)行剛剛添加的測(cè)試方法,測(cè)試通過,輸出日志如下:
DEBUG [main] - ==> Preparing: SELECT r.id, r.role_name roleName, r.enabled, r.create_by createBy, r.create_time createTime FROM sys_user u INNER JOIN sys_user_role ur ON u.id = ur.user_id INNER JOIN sys_role r ON ur.role_id = r.id WHERE u.id = ? AND r.enabled = ? DEBUG [main] - ==> Parameters: 1(Long), 1(Integer) TRACE [main] - <== Columns: id, roleName, enabled, createBy, createTime TRACE [main] - <== Row: 1, 管理員, 1, 1, 2019-06-27 18:21:12.0 TRACE [main] - <== Row: 2, 普通用戶, 1, 1, 2019-06-27 18:21:12.0 DEBUG [main] - <== Total: 2
2.2 參數(shù)類型是對(duì)象
為了演示參數(shù)類型是對(duì)象的使用方法,我們?cè)诮涌赟ysUserMapper中添加如下方法:
/** * 根據(jù)用戶id和角色的enabled狀態(tài)獲取用戶的角色 * * @param user * @param role * @return */ List<SysRole> selectRolesByUserAndRole(@Param("user") SysUser user, @Param("role") SysRole role);
此時(shí)對(duì)應(yīng)的xml中的語句為:
<select id="selectRolesByUserAndRole" resultType="com.zwwhnly.mybatisaction.model.SysRole"> SELECT r.id, r.role_name roleName, r.enabled, r.create_by createBy, r.create_time createTime FROM sys_user u INNER JOIN sys_user_role ur ON u.id = ur.user_id INNER JOIN sys_role r ON ur.role_id = r.id WHERE u.id = #{user.id} AND r.enabled = #{role.enabled} </select>
以上是“MyBatis中XML基本用法之多表查詢功能的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。