Mybatis基礎: 常見問題與FAQ

Mybatis基礎: #{...} 和 ${...} 的區別

MyBatis將 #{…} 解釋爲JDBC prepared statement 的一個參數標記。而將 ${…} 解釋爲字符串替換。理解這兩者的區別是很有用的, 因爲在某些SQL語句中並不能使用參數標記(parameter markers)。

比如,我們不能在表名(table name)的位置使用參數標記。
假設有下面的代碼:
[java] view plain copy
  1. Map<String, Object> parms = new HashMap<String, Object>();  
  2. parms.put("table""foo"); // 表名  
  3. parms.put("criteria"37); // 查詢過濾條件  
  4. List<Object> rows = mapper.generalSelect(parms);  
[html] view plain copy
  1. <select id="generalSelect" parameterType="map">  
  2.   select * from ${table} where col1 = #{criteria}  
  3. </select>  

MyBatis生成的SQL語句(prepared statement)如下所示:
[sql] view plain copy
  1. select * from foo where col1 = ?  

重要提示: 請注意,使用$ {…} (字符串替換)時可能會有SQL注入攻擊的風險。另外,字符串替換在處理複雜類型也可能常常發生問題,如日期類型。由於這些因素,我們建議您儘可能地使用 #{…} 這種方式。

要使用LIKE語句該怎麼寫?


有兩種使用LIKE的方法。(推薦使用)第一種方法是,在Java代碼中添加SQL通配符。

示例一:
[java] view plain copy
  1. String wildcardName = "%Smi%";  
  2. List<Name> names = mapper.selectLike(wildcardName);  
[html] view plain copy
  1. <select id="selectLike">  
  2.   select * from foo where bar like #{value}  
  3. </select>  

第二種方式是在SQL語句中拼接通配符。這種方法相對來說安全性要低一些,因爲可能會被SQL注入攻擊。
示例二:
[java] view plain copy
  1. String wildcardName = "Smi";  
  2. List<Name> names = mapper.selectLike(wildcardName);  
[html] view plain copy
  1. <select id="selectLike">  
  2.   select * from foo where bar like '%' || '${value}' || '%'  
  3. </select>  

重要提示: 請注意兩種方式中 $ 和 的使用!

如何執行批量插入?


首先,創建一個簡單的insert語句:
[html] view plain copy
  1. <insert id="insertName">  
  2.   insert into names (name) values (#{value})  
  3. </insert>  

然後在Java代碼中像下面這樣執行批處理插入:
[java] view plain copy
  1. List<String> names = new ArrayList<String>();  
  2. names.add("Fred");  
  3. names.add("Barney");  
  4. names.add("Betty");  
  5. names.add("Wilma");  
  6.   
  7. // 注意這裏 ExecutorType.BATCH  
  8. SqlSession sqlSession = sqlSessionFactory.openSession(ExecutorType.BATCH);  
  9. try {  
  10.   NameMapper mapper = sqlSession.getMapper(NameMapper.class);  
  11.   for (String name : names) {  
  12.     mapper.insertName(name);  
  13.   }  
  14.   sqlSession.commit();  
  15. finally {  
  16.   sqlSession.close();  
  17. }  

如何獲取自動生成的(主)鍵值?


insert 方法總是返回一個int值 - 這個值代表的是插入的行數。而自動生成的鍵值在 insert 方法執行完後可以被設置到傳入的參數對象中。
示例:
[html] view plain copy
  1. <insert id="insertName" useGeneratedKeys="true" keyProperty="id">  
  2.   insert into names (name) values (#{name})  
  3. </insert>  
[java] view plain copy
  1. Name name = new Name();  
  2. name.setName("Fred");  
  3.   
  4. int rows = mapper.insertName(name);  
  5. // 完成後,id已經被設置到對象中  
  6. System.out.println("rows inserted = " + rows);  
  7. System.out.println("generated key value = " + name.getId());  

在mapper中如何傳遞多個參數?

Java的反射機制並不能讓框架獲取到參數的名字(方法簽名中只有參數類型,可以說是爲了優化,也可以說設計就是如此,總之名字無意義), 所以MyBatis默認的命名爲: param1,param2……
如果想給他們指定名稱,可以使用 @param 註解:
[java] view plain copy
  1. import org.apache.ibatis.annotations.Param;  
  2. public interface UserMapper {  
  3.    User selectUser(@Param("username") String username,   
  4.                    @Param("hashedPassword") String hashedPassword);  
  5. }  

然後,就可以在xml像下面這樣使用(推薦封裝爲一個Map,作爲單個參數傳遞給Mapper):
[html] view plain copy
  1. <select id=”selectUser” resultType=”User”>  
  2.   select id, username, hashedPassword  
  3.   from some_table  
  4.   where username = #{username}  
  5.   and hashedPassword = #{hashedPassword}  
  6. </select>  

原文鏈接: http://blog.csdn.net/renfufei/article/details/39649707

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章