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

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|JavaScript|易語言|

服務(wù)器之家 - 編程語言 - Java教程 - 解析Mybatis判斷表達(dá)式源碼分析

解析Mybatis判斷表達(dá)式源碼分析

2021-06-04 13:29Enjoyjava Java教程

這篇文章主要介紹了Mybatis判斷表達(dá)式源碼分析,本文通過實例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下

在我們開發(fā)過程中用 mybatis 經(jīng)常會用到下面的例子

mapper如下

?
1
map<string ,string > testarray(@param("array") string [] array);

xml中的sql如下

?
1
2
3
4
5
6
<select id="testarray" resulttype="map">
  select * from t_ams_ac_pmt_dtl where cpt_pro=#{cptprop}
  <if test="array!=null and array != '' ">
    and cpt_pro=#{cptprop}
  </if>
</select>

剛看上面的代碼會覺得數(shù)組怎么能和空字符串進(jìn)行一起比較呢,一開始會覺得這個代碼運行起來絕對報錯,但是寫單元測試運行了一遍發(fā)現(xiàn)成功運行了。因此想是不是 mybatis 在內(nèi)部對數(shù)組類型的數(shù)據(jù)進(jìn)行了封裝。于是有了這一次的源碼解析之旅。上網(wǎng)查了查發(fā)現(xiàn) mybatis 解析使用了 ognl 。至于什么是 ognl 摘抄了百度百科中的一段話

ognl是object-graph navigation language的縮寫,它是一種功能強(qiáng)大的表達(dá)式語言,通過它簡單一致的表達(dá)式語法,可以存取對象的任意屬性,調(diào)用對象的方法,遍歷整個對象的結(jié)構(gòu)圖,實現(xiàn)字段類型轉(zhuǎn)化等功能。它使用相同的表達(dá)式去存取對象的屬性。這樣可以更好的取得數(shù)據(jù)。

單元測試類如下

?
1
2
3
4
5
6
7
8
@test
  public void testarray(){
    sqlsession sqlsession = sqlsessionfactory.opensession();
    tbapcheckptstranscdmapper mapper = sqlsession.getmapper(tbapcheckptstranscdmapper.class);
    string str= "1,2,3";
    string [] strings = str.split(",");
    mapper.testarray(strings);
  }

首先我們先來看一下 dynamicsqlsource 這個類,這個類中有個方法如下

?
1
2
3
4
5
6
7
8
9
10
11
12
13
@override
 public boundsql getboundsql(object parameterobject) {
  dynamiccontext context = new dynamiccontext(configuration, parameterobject);
  rootsqlnode.apply(context);
  sqlsourcebuilder sqlsourceparser = new sqlsourcebuilder(configuration);
  class<?> parametertype = parameterobject == null ? object.class : parameterobject.getclass();
  sqlsource sqlsource = sqlsourceparser.parse(context.getsql(), parametertype, context.getbindings());
  boundsql boundsql = sqlsource.getboundsql(parameterobject);
  for (map.entry<string, object> entry : context.getbindings().entryset()) {
   boundsql.setadditionalparameter(entry.getkey(), entry.getvalue());
  }
  return boundsql;
 }

其中

rootsqlnode.apply(context);

這段代碼對sql進(jìn)行了動態(tài)的拼接,然后點進(jìn)去看一下

?
1
2
3
4
5
6
7
@override
 public boolean apply(dynamiccontext context) {
  for (sqlnode sqlnode : contents) {
   sqlnode.apply(context);
  }
  return true;
 }

這里的sql拼接運用了 組合模式 不同的 sqlnode 調(diào)用的方法不一樣,但是最后的想要結(jié)果都是一樣的:拼接sql。例如我們第一次進(jìn) apply 這個方法中的時候他跳轉(zhuǎn)到了

statictextsqlnode 這個類中調(diào)用了下面的方法

?
1
2
3
4
5
@override
 public boolean apply(dynamiccontext context) {
  context.appendsql(text);
  return true;
 }

直接將sql拼接為

?
1
select * from t_ams_ac_pmt_dtl where cpt_pro=#{cptprop}

然后我們第二次循環(huán)執(zhí)行發(fā)現(xiàn)它跳轉(zhuǎn)到了 ifsqlnode 這個類中,這是標(biāo)簽為 <if> 的判斷類,

?
1
2
3
4
5
6
7
8
@override
 public boolean apply(dynamiccontext context) {
  if (evaluator.evaluateboolean(test, context.getbindings())) {
   contents.apply(context);
   return true;
  }
  return false;
 }

在解析語句中傳了兩個參數(shù)進(jìn)去

?
1
evaluator.evaluateboolean(test, context.getbindings())

test :就是要解析的表達(dá)式,在此場景下就是 array!=null and array != ''
context.getbindings() :獲得的是一個map,其中存儲了參數(shù) array 的所對應(yīng)的值,如下所示

image

然后接下來就到了 ognl 解析表達(dá)式了,發(fā)現(xiàn)最后到了 astnoteq 這類中

?
1
2
3
4
5
protected object getvaluebody(ognlcontext context, object source) throws ognlexception {
    object v1 = this._children[0].getvalue(context, source);
    object v2 = this._children[1].getvalue(context, source);
    return ognlops.equal(v1, v2) ? boolean.false : boolean.true;
  }

這里解析分為了兩步進(jìn)行解析,上面的表達(dá)式為 array!=null and array != '' 那么他會根據(jù)and 進(jìn)行分組將其放入 node 數(shù)組中。

?
1
2
node[0] : array!=null
node[1] : array != ''

然后這里面的兩個參數(shù) v1 和 v2 分別為左邊和右邊的參數(shù),此時先解析 node[0] 中的參數(shù)

  • v1 :就是參數(shù) array 對應(yīng)的數(shù)組的值
  • v2 :就是null

此時到這應(yīng)該就知道為什么 string 數(shù)組為什么能和空字符串進(jìn)行比較了,因為他將數(shù)組轉(zhuǎn)化為了 object 然后用自己寫的 equal 方法進(jìn)行比較。然后進(jìn)去他寫的 equal 方法中看了以后發(fā)現(xiàn)他對數(shù)組比較是特殊的。

  • 如果左邊是數(shù)組右邊是字符串:兩個都轉(zhuǎn)換為 object 然后進(jìn)行 v1.getclass()==v2.getclass() 判斷
  • 如果左邊是數(shù)組右邊也是數(shù)組:先判斷兩個數(shù)組的長度是否相同,如果相同,那么循環(huán)遍歷兩個數(shù)組進(jìn)行里面的值的比較

總結(jié)

以上所述是小編給大家介紹的解析mybatis判斷表達(dá)式源碼分析,希望對大家有所幫助,如果大家有任何歡迎給我留言,小編會及時回復(fù)大家的!

原文鏈接:https://studygolang.com/articles/15449

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 边摸边吃奶边做爽视频免费 | 草莓视频旧版 | 7777奇米影视| 亚洲欧美日韩综合一区久久 | 99久久一香蕉国产线看观看 | 调教全程肉动画片在线观看 | 无套插入| 国产亚洲99影院 | 91免费视频国产 | 国产成人亚洲综合91精品555 | 国产精品久久久久jk制服 | 吃瓜视频在线观看 | 91aaa免费免费国产在线观看 | 俄罗斯男男激情1069gay | 天天久久综合网站 | 厨房里摸着乳丰满在线观看 | 色综合久久中文字幕综合网 | 娇妻与公陈峰姚瑶最新版 | 久久婷婷五月免费综合色啪 | www.大逼色 | 成人欧美一区二区三区白人 | 国产午夜永久福利视频在线观看 | 四虎在线视频免费观看 | 亚洲一区二区精品推荐 | 午夜想想爱午夜剧场 | 美女国内精品自产拍在线播放 | 国产主播99 | 欧美日韩国产亚洲人成 | 日韩精品一区二区三区中文字幕 | hezyo加勒比一区二区三区 | 69日本人 | 果冻传媒天美传媒网址入口 | www.国产在线观看 | 四虎2020紧急免费入口 | 免费国产午夜高清在线视频 | 黑帮大佬与我的365天2标清中文 | 亚洲七七久久综合桃花 | 男女真实无遮挡xx00动态图软件 | 4444www免费看 | 亚洲精品乱码久久久久久蜜桃 | 九九九久久久 |