一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - mybatis查詢語句揭秘之封裝數據

mybatis查詢語句揭秘之封裝數據

2021-07-29 11:31不懂是非 Java教程

這篇文章主要給大家介紹了關于mybatis查詢語句揭秘之封裝數據的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用mybatis具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧

一、前言

繼上一篇mybatis查詢語句的背后,這一篇主要圍繞著mybatis查詢的后期操作,即跟數據庫交互的時候。由于本人也是一邊學習源碼一邊記錄,內容難免有錯誤或不足之處,還望諸位指正,本文只可當參考作用。謹記!

二、分析

繼上一篇博文的查詢例子,mybatis在最后的查詢最終會走simpleexecutor類的doquery方法,

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
@override
 public <e> list<e> doquery(mappedstatement ms, object parameter, rowbounds rowbounds, resulthandler resulthandler, boundsql boundsql) throws sqlexception {
 statement stmt = null;
 try {
 configuration configuration = ms.getconfiguration();
 // 這里也就是采用了策略模式(個人感覺有點像),實際的statementhandler為routingstatementhandler
 statementhandler handler = configuration.newstatementhandler(wrapper, ms, parameter, rowbounds, resulthandler, boundsql);
 stmt = preparestatement(handler, ms.getstatementlog());
 // 雖然是執行的routingstatementhandler.query,但返回結果的還是preparedstatementhandler處理
 return handler.query(stmt, resulthandler);
 } finally {
 closestatement(stmt);
 }
 }
 
private statement preparestatement(statementhandler handler, log statementlog) throws sqlexception {
 statement stmt;
 // 使用了代理模式,也可以理解為對connection進行了一層包裝,這里的作用就是加了log處理
 connection connection = getconnection(statementlog);
 //進行預編譯,即類似jdbc的 sql,如 select * from user where id=?
 stmt = handler.prepare(connection, transaction.gettimeout());
 // 對執行查詢的sql進行參數設置
 handler.parameterize(stmt);
 return stmt;
 }

關于 handler.prepare的作用這里簡單介紹下,不做代碼分析。

會設置fetchsize,作用就是一次性從數據庫抓取數據,好像默認值是10條,如果每次只抓取一條,則進行rs.next的時候,會再次查庫。

如果是insert操作,并且數據庫主鍵自增且還設置了可以返回主鍵,則會還做獲取主鍵的操作。

先從設置參數說起,也就是handler.parameterize。先看下源碼,具體位置在defaultparameterhandler類里面

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
@override
 public void setparameters(preparedstatement ps) {
 errorcontext.instance().activity("setting parameters").object(mappedstatement.getparametermap().getid());
 // 獲取配置文件里面的sql參數信息,如sql為select * from user where id=#{userid,jdbctype=integer}
 // parametermapping 記錄了參數名也就是userid,還有記錄了對應的jdbc類型,還有對應的javatype等等,具體可以debug看下
 list<parametermapping> parametermappings = boundsql.getparametermappings();
 if (parametermappings != null) {
 for (int i = 0; i < parametermappings.size(); i++) {
 parametermapping parametermapping = parametermappings.get(i);
 if (parametermapping.getmode() != parametermode.out) {
  object value;
  string propertyname = parametermapping.getproperty();
  // 如果為true,那么sql參數中有類似 user.name 格式
  if (boundsql.hasadditionalparameter(propertyname)) { // issue #448 ask first for additional params
  value = boundsql.getadditionalparameter(propertyname);
  } else if (parameterobject == null) {
  value = null;
  } else if (typehandlerregistry.hastypehandler(parameterobject.getclass())) {
  value = parameterobject;
  } else {
  // metaobject 類似一個工具類,它里面有一個反射工廠,可以專門解析一個類的信息,如字段的setter/getter/屬性信息,這里不做多余介紹
  // 1、下面詳細介紹
  metaobject metaobject = configuration.newmetaobject(parameterobject);
  value = metaobject.getvalue(propertyname);// 取值
  }
  // 獲取對應的typehandler,一般情況不設置的話,基本都是objecttypehandler
  typehandler typehandler = parametermapping.gettypehandler();
  jdbctype jdbctype = parametermapping.getjdbctype();
  if (value == null && jdbctype == null) {
  jdbctype = configuration.getjdbctypefornull();
  }
  try {
  // 進行設值
  typehandler.setparameter(ps, i + 1, value, jdbctype);
  } catch (typeexception e) {
  throw new typeexception("could not set parameters for mapping: " + parametermapping + ". cause: " + e, e);
  } catch (sqlexception e) {
  throw new typeexception("could not set parameters for mapping: " + parametermapping + ". cause: " + e, e);
  }
 }
 }
 }
 }

對于上述代碼中的一部分這里負責將parameterobject的里面的值整出來(也就是傳入的參數),如果參數是map結構,就從map里面取值,如果不是,如單個非javabean參數,則直接取值,如果是單個javabean,則通過metaobject類轉換成一個beanwrapper,進行取值

這段代碼也就負責對預編譯后的sql設置參數,這里邏輯主要是圍繞以下步驟進行得,

獲取參數名,獲取參數值,獲取參數類型,然后做進行設值操作

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
/**
 * mybatis數據處理有單結果集和多結果集處理,一般多結果集出現存儲過程中,如果存儲過程中寫了兩條select語句,如
 * select * from user , select * from classes 這種情況這里不做介紹,因為本人用的不多,理解的也不是很透徹。
 * 這里不多做介紹,這里只針對簡單映射做一個大概介紹
 *
 */
public list<object> handleresultsets(statement stmt) throws sqlexception {
 errorcontext.instance().activity("handling results").object(mappedstatement.getid());
 // 保存查詢結果
 final list<object> multipleresults = new arraylist<>();
 
 int resultsetcount = 0;
 // 獲取第一條數據
 resultsetwrapper rsw = getfirstresultset(stmt);
 // 如果不是多結果集映射,一般resultmaps的大小為1
 // resultmap中存儲的有類的字段屬性,數據庫字段名稱等信息
 list<resultmap> resultmaps = mappedstatement.getresultmaps();
 int resultmapcount = resultmaps.size();
 // 校驗數據的正確性
 validateresultmapscount(rsw, resultmapcount);
 while (rsw != null && resultmapcount > resultsetcount) {
 resultmap resultmap = resultmaps.get(resultsetcount);
 // 處理結果集映射
 handleresultset(rsw, resultmap, multipleresults, null);
 rsw = getnextresultset(stmt);
 cleanupafterhandlingresultset();
 resultsetcount++;
 }
 // 處理slect 標簽的resultsets屬性,多個用逗號隔開,個人幾乎沒用過,略過
 string[] resultsets = mappedstatement.getresultsets();
 if (resultsets != null) {
 while (rsw != null && resultsetcount < resultsets.length) {
 resultmapping parentmapping = nextresultmaps.get(resultsets[resultsetcount]);
 if (parentmapping != null) {
  string nestedresultmapid = parentmapping.getnestedresultmapid();
  resultmap resultmap = configuration.getresultmap(nestedresultmapid);
  handleresultset(rsw, resultmap, null, parentmapping);
 }
 rsw = getnextresultset(stmt);
 cleanupafterhandlingresultset();
 resultsetcount++;
 }
 }
 
 return collapsesingleresultlist(multipleresults);
 }

以上代碼就是為結果映射做一個鋪墊,重點是在hanleresultset方法里,

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
private void handleresultset(resultsetwrapper rsw, resultmap resultmap, list<object> multipleresults, resultmapping parentmapping) throws sqlexception {
 try {// 針對簡單映射,parentmapping是為null的
 if (parentmapping != null) {
 handlerowvalues(rsw, resultmap, null, rowbounds.default, parentmapping);
 } else {
 // 默認使用defaultresulthandler,如需使用自定義的,則可在傳參加入resulthandler接口實現類
 if (resulthandler == null) {
  defaultresulthandler defaultresulthandler = new defaultresulthandler(objectfactory);
  // 處理結果,結果存在resulthandler里
  handlerowvalues(rsw, resultmap, defaultresulthandler, rowbounds, null);
  multipleresults.add(defaultresulthandler.getresultlist());
 } else {
  handlerowvalues(rsw, resultmap, resulthandler, rowbounds, null);
 }
 }
 } finally {
 // issue #228 (close resultsets)
 closeresultset(rsw.getresultset());
 }
 }
?
1
2
3
4
5
6
7
8
9
10
public void handlerowvalues(resultsetwrapper rsw, resultmap resultmap, resulthandler<?> resulthandler, rowbounds rowbounds, resultmapping parentmapping) throws sqlexception {
 // 處理有嵌套映射的情況
 if (resultmap.hasnestedresultmaps()) {
 ensurenorowbounds();
 checkresulthandler();
 handlerowvaluesfornestedresultmap(rsw, resultmap, resulthandler, rowbounds, parentmapping);
 } else {//沒有嵌套映射
 handlerowvaluesforsimpleresultmap(rsw, resultmap, resulthandler, rowbounds, parentmapping);
 }
 }
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
private void handlerowvaluesforsimpleresultmap(resultsetwrapper rsw, resultmap resultmap, resulthandler<?> resulthandler, rowbounds rowbounds, resultmapping parentmapping)
 throws sqlexception {
 defaultresultcontext<object> resultcontext = new defaultresultcontext<>();
 resultset resultset = rsw.getresultset();
 // 跳過多少行,到達指定記錄位置,如在傳參的時候傳入了rowbounds,則會根據該類的offset值跳到指定記錄位置
 skiprows(resultset, rowbounds);
 // shouldprocessmorerows 用來檢測是否能繼續對后續的結果進行映射
 while (shouldprocessmorerows(resultcontext, rowbounds) && !resultset.isclosed() && resultset.next()) {
 //用來處理resultmap節點中配置了discriminator節點,這里忽略掉
 resultmap discriminatedresultmap = resolvediscriminatedresultmap(resultset, resultmap, null);
 // 得到的結果就是sql執行后的一行記錄,如返回user對象信息,則rowvalue就代表一個user實例,里面已經有值了
 object rowvalue = getrowvalue(rsw, discriminatedresultmap, null);
 //保存數據
 storeobject(resulthandler, resultcontext, rowvalue, parentmapping, resultset);
 }
 }
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
private object getrowvalue(resultsetwrapper rsw, resultmap resultmap, string columnprefix) throws sqlexception {
 final resultloadermap lazyloader = new resultloadermap();
 // 創建對象,可以理解為對resultmap節點的type屬性值,進行了反射處理,得到了一個對象,但屬性值都是默認值。
 object rowvalue = createresultobject(rsw, resultmap, lazyloader, columnprefix);
 if (rowvalue != null && !hastypehandlerforresultobject(rsw, resultmap.gettype())) {
 final metaobject metaobject = configuration.newmetaobject(rowvalue);
 boolean foundvalues = this.useconstructormappings;
 //是否需要自動映射,有三種映射,分別為none,partial,full,默認第二種,處理非嵌套映射,可通過automappingbehavior 配置
 if (shouldapplyautomaticmappings(resultmap, false)) {
 // 映射resultmap中未明確指定的列,如類中含有username屬性,但是resultmap中沒配置,則通過這個進行數據映射,還是可以查詢到結果
 foundvalues = applyautomaticmappings(rsw, resultmap, metaobject, columnprefix) || foundvalues;
 }
 // 處理resultmap中指定的列
 foundvalues = applypropertymappings(rsw, resultmap, metaobject, lazyloader, columnprefix) || foundvalues;
 foundvalues = lazyloader.size() > 0 || foundvalues;
 // 如果沒查詢到結果,但配置可返回空對象(指的是沒有設置屬性值得對象),則返回空對象,否則返回null
 rowvalue = foundvalues || configuration.isreturninstanceforemptyrow() ? rowvalue : null;
 }
 return rowvalue;
 }

這里只介紹resultmap中有明確指定的列

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
private boolean applypropertymappings(resultsetwrapper rsw, resultmap resultmap, metaobject metaobject, resultloadermap lazyloader, string columnprefix)
 throws sqlexception {
 // 獲取數據字段名
 final list<string> mappedcolumnnames = rsw.getmappedcolumnnames(resultmap, columnprefix);
 boolean foundvalues = false;
 // 獲取的數據就是resultmap節點中配置的result節點,有多個result節點,這個集合大小就是多少
 // 里面存儲的是屬性名/字段名等信息
 final list<resultmapping> propertymappings = resultmap.getpropertyresultmappings();
 for (resultmapping propertymapping : propertymappings) {
 string column = prependprefix(propertymapping.getcolumn(), columnprefix);
 // 是否有嵌套映射
 if (propertymapping.getnestedresultmapid() != null) {
 // the user added a column attribute to a nested result map, ignore it
 column = null;
 }
 // 針對1來說一般常與嵌套查詢配合使用
 // 2 判斷屬性基本映射
 // 3 多結果集的一個處理
 if (propertymapping.iscompositeresult()// 1
  || (column != null && mappedcolumnnames.contains(column.touppercase(locale.english)))// 2
  || propertymapping.getresultset() != null) {// 3
 // 獲取當前column字段對于的值,有用到typehandler來進行參數的一個轉換
 object value = getpropertymappingvalue(rsw.getresultset(), metaobject, propertymapping, lazyloader, columnprefix);
 
 //獲取類的屬性字段名
 final string property = propertymapping.getproperty();
 if (property == null) {
  continue;
 } else if (value == deferred) {// 類似占位符。處理懶加載數據
  foundvalues = true;
  continue;
 }
 if (value != null) {
  foundvalues = true;
 }
 if (value != null || (configuration.iscallsettersonnulls() && !metaobject.getsettertype(property).isprimitive())) {
  // 進行設置屬性值
  metaobject.setvalue(property, value);
 }
 }
 }
 return foundvalues;
 }

或許有人奇怪為啥沒看到查詢的對象有set操作,值就到了對象里面去了,這里全是metaobject給你操作了,具體的,大家可以自行了解這個類,只能說這個類的功能很強大。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對服務器之家的支持。

原文鏈接:https://www.cnblogs.com/qm-article/p/10588627.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 小草观看免费高清视频 | 黑帮大佬与我的365天2标清中文 | 国产精品麻豆 | 包臀裙女教师波多野结衣 | 成年人视频在线免费观看 | 美女的让男人桶爽网站 | 欧美成人手机 | 高清在线免费 | 护士让我吃奶我扒她奶 | 四神集团1涨奶是第几章 | 免费看视频高清在线观看 | jzz大全部| 免费标准高清看机机桶机机 | 国产成人一区二区三区在线视频 | 日本一区二区三区久久 | 免费人成黄页在线观看69 | 免费亚洲视频 | 国产资源在线视频 | 97久久天天综合色天天综合色hd | 日韩一区二区三区四区区区 | 色天使亚洲综合在线观看 | 色婷婷影院在线视频免费播放 | 国模娜娜a4u1546全套 | 秋霞啪啪片 | 欧美精品一区二区三区久久 | 欧美一级久久久久久久大片 | 免费精品在线视频 | 亚洲男人天堂 | 久久国产精品永久免费网站 | 色综合天天综合网国产人 | 日本中文字幕在线视频 | 特黄aa级毛片免费视频播放 | 国产1区2区三区不卡 | 出差上的少妇20p | 思思91精品国产综合在线 | 34看网片午夜理 | 国产午夜视频在线观看网站 | 强行扒开美女大腿挺进 | 日本高清在线不卡 | 邪恶肉肉全彩色无遮盖 | 99re在线精品视频免费 |