簡(jiǎn)單查詢-resultType
我們提供的服務(wù)有:做網(wǎng)站、網(wǎng)站制作、微信公眾號(hào)開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認(rèn)證、南開ssl等。為數(shù)千家企事業(yè)單位解決了網(wǎng)站和推廣的問題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的南開網(wǎng)站制作公司
數(shù)據(jù)準(zhǔn)備
字段 | 注釋 |
---|---|
SNO | 學(xué)號(hào) |
SNAME | 學(xué)生名字 |
SSEX | 性別 |
SBIRITHDAY | 生日 |
CLASS | 班級(jí) |
create table TEST.STUDENT ( SNO varchar(3) not null, SNAME varchar(4) not null, SSEX varchar(2) not null, SBIRTHDAY datetime null, CLASS varchar(5) null )
public class Student { private String SNO; private String SNAME; private String SSEX; private Date SBIRTHDAY; private String CLASS; ... }
例子
按照返回?cái)?shù)據(jù)類型大致分為基礎(chǔ)數(shù)據(jù)類型,JavaBean 和 Map。其中雖然返回的結(jié)果行數(shù)有單條也有多條,對(duì)應(yīng)的接口返回類型是集合或者單個(gè)對(duì)象,但是在 xml 映射文件中,resultType 的值是相同的。
1、指定字段-基礎(chǔ)數(shù)據(jù)類型
接口類:
String querySingleStudent(); ListqueryAllStudent();
Mapper 文件:
2、Map,一般為 hashMap
接口類:
MapqueryStudentMap(); List
其中:
3、JavaBean
接口類:
Student querySingleStudentBean(); ListqueryAllStudentBean();
Mapper 文件:
resultType="student" 為 Student.java 的別名,也可以是全限定名。別名在 mybatis-config.xml 配置文件中設(shè)置:
...
但是如果 JavaBean 文件很多,不想一個(gè)個(gè)指定,也可以使用 package 標(biāo)簽 設(shè)置mybatis自動(dòng)掃描,別名即為類名的小寫。
復(fù)雜查詢 resultMap
對(duì)于一般的查詢語(yǔ)句,resultType 足夠了。對(duì)于多表查詢等情況,就要請(qǐng)出 resultMap 了。
數(shù)據(jù)庫(kù)字段和 java 數(shù)據(jù)類型映射關(guān)系
數(shù)據(jù)庫(kù)字段類型 jdbcType 和 java 數(shù)據(jù)類型 并不是一一對(duì)應(yīng)的關(guān)系,而且不同數(shù)據(jù)庫(kù)類型也不盡相同。而 mybatis 將 TypeHandler 作為兩者之間的映射關(guān)系。大部分情況下都是沒有問題的,但是并非能覆蓋所有的情況,特殊情況下可以使用 resultMap 自定義這種映射關(guān)系。
舉個(gè)例子,數(shù)據(jù)庫(kù) LongVarchar 字段類型對(duì)應(yīng) java 中的 String 類型。但是在 DB2 數(shù)據(jù)庫(kù)中,查詢的 LongVarchar 類型的字段,在 mybatis 中被識(shí)別成 jdbcType 為 BLOB。有兩種解決方法,第一種是在 SQL 中對(duì)該字段使用 CAST 轉(zhuǎn)換為 VARCHAR(長(zhǎng)度)類型。另一種是使用 resultMap:
1、
2、
3、
id
字段的映射關(guān)系的標(biāo)簽即有,也有,在 mybatis 文檔中指出不使用id,會(huì)造成性能下降,因此將主鍵字段使用 id 標(biāo)簽是推薦的做法。但是如果不存在主鍵呢,當(dāng)你在 ResultMap 只提供了部分字段而不是全部字段,即使使用了 autoMapping 屬性,那么 mybatis 會(huì)按照你提供的字段名進(jìn)行去重。那么在使用 resultMap 的時(shí)候,最優(yōu)選擇是:
多表關(guān)聯(lián)查詢
在 resultType 的例子中都只涉及到一張表,如果涉及多張表關(guān)聯(lián)查詢呢。我們可以簡(jiǎn)單的將所有列映射到 hashmap 的鍵值上。
但是 HashMap 不是一個(gè)很好的領(lǐng)域模型。 你的程序更可能會(huì)使用 JavaBean 或 POJO(Plain Old Java Objects,普通 Java 對(duì)象)作為領(lǐng)域模型。
因此這里均采用 JavaBean 作為領(lǐng)域模型。增加一個(gè)成績(jī)表 Score
字段 | 注釋 |
---|---|
SNO | 學(xué)號(hào) |
CNO | 課程編號(hào) |
DEGREE | 成績(jī) |
create table SCORE ( SNO varchar(3) not null, CNO varchar(5) not null, DEGREE decimal(10, 1) not null )
public class Score { private String SNO; private String CNO; private Double DEGREE; ... }
一對(duì)一關(guān)系
這里的一對(duì)多關(guān)系是兩個(gè)表字段一一對(duì)應(yīng),一個(gè)學(xué)生的某門課的成績(jī)是唯一確定的。 在一一對(duì)應(yīng)的情況下要在 resultMap 中使用 標(biāo)簽。
在 Student.java 中增加字段 Score
private Score score; public Score getScore() { return score; } public void setScore(Score score) { this.score = score; }
有兩種使用情況,第一種為嵌套查詢,即前一個(gè) SQL 查詢結(jié)果集中的字段作為參數(shù)傳遞給下一個(gè) SQL。第二種情況為嵌套結(jié)果集,即兩個(gè)表做關(guān)聯(lián)查詢,將結(jié)果集映射到多個(gè) JavaBean 文件。
嵌套查詢
select SNO,SNAME from test.STUDENT select degree from test.SCORE where sno = #{sno}
在標(biāo)簽中
另外需要注意的是這種嵌套查詢對(duì)于大型結(jié)果集和列名并友好,存在 N+1 的問題,因?yàn)橄乱粭l SQL 會(huì)執(zhí)行 N 次去循環(huán)查詢,使用關(guān)聯(lián)查詢更合適。再者也可以開啟 mybatis 的懶查詢功能,嵌套的 SQL 不是一口氣順序執(zhí)行完,而是在使用的時(shí)候才會(huì)執(zhí)行下一條 SQL。例如執(zhí)行student.getScore().getSNO()才會(huì)執(zhí)行queryScore的 SQL。默認(rèn)情況下沒有開啟,需要在配置文件中設(shè)置
設(shè)置參數(shù) | 描述 | 默認(rèn)值 |
---|---|---|
lazyLoadingEnabled | 延遲加載的全局開關(guān),特定關(guān)聯(lián)關(guān)系中可通過設(shè)置 fetchType 屬性來(lái)覆蓋該項(xiàng)的開關(guān)狀態(tài) | false |
aggressiveLazyLoading | 當(dāng)開啟時(shí),任何方法的調(diào)用都會(huì)加載該對(duì)象的所有屬性。否則,每個(gè)屬性會(huì)按需加載 | false (true in ≤3.4.1) |
也可以在標(biāo)簽中設(shè)置 fetchType = “l(fā)azy” 開啟懶加載,會(huì)覆蓋全局的參數(shù)設(shè)置。
嵌套結(jié)果集
對(duì)于多表關(guān)聯(lián)查詢,一般在 SQL 中使用別名來(lái)避免字段名的重復(fù)。mybatis 要做的是將別名正確的映射到 JavaBean 屬性上。
SELECT SNAME, SSEX, CLASS, ST.SNO, SC.SNO AS SC_SNO FROM test.student st INNER JOIN test.score sc ON st.sno = sc.sno where CNO = '3-105';
通過設(shè)置標(biāo)簽指定了表列名和屬性之間的映射關(guān)系。但這樣如果字段很多,會(huì)需要一一指定,標(biāo)簽提供了columnPrefix屬性,指定別名的前綴,這樣可以重用resultMap
一對(duì)多關(guān)系
除了一對(duì)一的關(guān)系,還有一對(duì)多的關(guān)系,比如這里一個(gè)學(xué)生Student 對(duì)應(yīng)多門課的成績(jī)。 一對(duì)多對(duì)應(yīng)的情況下要在 resultMap 中使用 標(biāo)簽。首先需要調(diào)整 JavaBean 文件中兩個(gè)表之間的關(guān)系。
private Listscore; public List getScore() { return score; } public void setScore(List score) { this.score = score; }
以嵌套結(jié)果集為例
SELECT SNAME,SSEX,CLASS,ST.SNO,SC.SNO AS SC_SNO FROM test.student st INNER JOIN test.score sc ON st.sno = sc.sno
注意到相比 association 多了一個(gè)屬性ofType,是用來(lái)表示 List 集合中的類型的。其他屬性的用法同 association 是一樣的。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)創(chuàng)新互聯(lián)的支持。