mybatis中的變量#與$


ibatis中使用select top #num# * from tableName出現錯誤。由於初次用ibatis還不知道在它里邊拼寫SQL語句的一些規則,導致一些自認為很平常的SQL語句,在它這里翻了船。

select top #number#這種寫法是不正確的,原因待查。正確的書寫方式是 select top $number$ 。

下面這段話是在網絡里找到的它也沒給出具體的解釋,只是說這是什么SQL的動態寫法,不明所以。

在iBATIS中,對於top參數,只能用動態SQL方式。如
<select id="getClassLatest" parameterClass="map" resultMap="musicItemMap">
   SELECT top $number$ * FROM tbl_music_item where class_id=#classId# order by add_time desc
</select>

#與$的區別

 

昨天一個項目中在寫ibatis中的sql語句時,order by #field#, 運行時總是報錯,后來上網查了查,才知道這里不該用#,而應該用$,隨即查了下#與$的區別. 
總結如下: 
1.#是把傳入的數據當作字符串,如#field#傳入的是id,則sql語句生成是這樣,order by "id",這當然會報錯.. 

2.$傳入的數據直接生成在sql里,如$field$傳入的是id,則sql語句生成是這樣,order by id, 這就對了. 

3.#方式能夠很大程度防止sql注入. 

4.$方式無法防止sql注入. 

5.$方式一般用於傳入數據庫對象.例如傳入表名. 

6.一般能用#的就別用$. 

MyBatis傳入參數與parameterType

Mybatis的Mapper文件中的select、insert、update、delete元素中有一個parameterType屬性,用於對應的mapper接口方法接受的參數類型。

可以接受的參數類型有基本類型和復雜類型。

mapper接口方法一般接受一個參數,可以通過使用@Param注釋將多個參數綁定到一個map做為輸入參數。

    1. 簡單數據類型

mapper接口方法:

User selectByPrimaryKey(Integer id);

sql映射:

<select id="selectByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.Integer" >
  select
  <include refid="Base_Column_List" />
   from base.tb_user
  where id = #{id,jdbcType=INTEGER}
</select>

對於簡單數據類型,sql映射語句中直接#{變量名}這種方式引用就行了,其實這里的”變量名”可以是任意的。mapper接口方法傳遞過來的值,至於其叫什么名字其實是不可考也沒必要知道的。
而且JAVA反射只能獲取方法參數的類型,是無從得知方法參數的名字的。

比如上面這個示例中,使用#{id}來引用只是比較直觀而已,使用其他名字來引用也是一樣的。所以當在if元素中test傳遞的參數時,就必須要用_parameter來引用這個參數了。像這樣:

<select id="selectByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.Integer" >
  select
  <include refid="Base_Column_List" />
  from tb_user
  <if test="_parameter != 0">
  where id = #{id,jdbcType=INTEGER}
  </if>
</select>

如果test測試條件中使用id就會提示錯誤,因為這個參數其實沒有名字,只是一個值或引用而已,只能使用_parameter來引用。

    1. 對象類型

傳入JAVA復雜對象類型的話,sql映射語句中就可以直接引用對象的屬性名了,這里的屬性名是實實在在的真實的名字,不是隨意指定的。
mapper接口方法:

int insert(User user);

sql映射:

<insert id="insert" parameterType="User" useGeneratedKeys="true" keyProperty="id">
  insert into tb_user (name, sex)
  values (#{name,jdbcType=CHAR}, #{sex,jdbcType=CHAR})

雖然可以明確的引用對象的屬性名了,但如果要在if元素中測試傳入的user參數,仍然要使用_parameter來引用傳遞進來的實際參數,因為傳遞進來的User對象的名字是不可考的。如果測試對象的屬性,則直接引用屬性名字就可以了。

測試user對象:

<if test="_parameter != null">

測試user對象的屬性:

<if test="name != null">
    1. map類型

傳入map類型,直接通過#{keyname}就可以引用到鍵對應的值。使用@param注釋的多個參數值也會組裝成一個map數據結構,和直接傳遞map進來沒有區別。

mapper接口:

int updateByExample(@Param("user") User user, @Param("example") UserExample example);

sql映射:

<update id="updateByExample" parameterType="map" >
  update tb_user
  set id = #{user.id,jdbcType=INTEGER},
  ...
  <if test="_parameter != null" >
    <include refid="Update_By_Example_Where_Clause" />
  </if>

注意這里測試傳遞進來的map是否為空,仍然使用_parameter

    1. 集合類型

You can pass a List instance or an Array to MyBatis as a parameter object. When you do, MyBatis will automatically wrap it in a Map, and key it by name. List instances will be keyed to the name “list” and array instances will be keyed to the name “array”.

可以傳遞一個List或Array類型的對象作為參數,MyBatis會自動的將List或Array對象包裝到一個Map對象中,List類型對象會使用list作為鍵名,而Array對象會用array作為鍵名。

集合類型通常用於構造IN條件,sql映射文件中使用foreach元素來遍歷List或Array元素。

mapper接口:

User selectUserInList(List<Interger> idlist);

sql動態語句映射:

<select id="selectUserInList" resultType="User">
  SELECT *
  FROM USER
  WHERE ID in
  <foreach item="item" index="index" collection="list"
      open="(" separator="," close=")">
        #{item}
  </foreach>
</select>
    1. 對象類型中的集合屬性

對於單獨傳遞的List或Array,在SQL映射文件中映射時,只能通過list或array來引用。但是如果對象類型有屬性的類型為List或Array,則在sql映射文件的foreach元素中,可以直接使用屬性名字來引用。
mapper接口:

List<User> selectByExample(UserExample example);

sql映射文件:

<where >
  <foreach collection="oredCriteria" item="criteria" separator="or" >
    <if test="criteria.valid" >

在這里,UserExample有一個屬性叫oredCriteria,其類型為List,所以在foreach元素里直接用屬性名oredCriteria引用這個List即可。

item=”criteria”表示使用criteria這個名字引用每一個集合中的每一個List或Array元素

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM