同事在學(xué)mybatis時(shí),遇到了一個(gè)問題就是,使用char類型字段作為查詢條件時(shí)一直都查不出數(shù)據(jù),其他類型的則可以。
站在用戶的角度思考問題,與客戶深入溝通,找到和平網(wǎng)站設(shè)計(jì)與和平網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個(gè)性化、用戶體驗(yàn)好的作品,建站類型包括:成都網(wǎng)站制作、做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、空間域名、虛擬主機(jī)、企業(yè)郵箱。業(yè)務(wù)覆蓋和平地區(qū)。
使用的數(shù)據(jù)庫是oracle,查詢條件字段類型是char(50),java代碼對應(yīng)的是String類型。
后來經(jīng)過排查,是由于在oracle中,char類型字段,如果內(nèi)容長度不夠,會自動以空格方式補(bǔ)足長度。如字段 name char(5),若值為sgl,那么oracle會自動用空格補(bǔ)足長度,最終值為sgl。
一、解決方法:
方法1:先用trim()函數(shù)把值去掉兩邊空格再作為條件查詢,如:
select * from data where data.name=#{name}
改為:
select * from data where trim(data.name)=#{name}
方法2:將字段類型char()改為varchar2()類型。一般情況下,只有所有值長度都一樣時(shí)才用char()類型,比如性別字段,用0表示男和1表示女時(shí),就可以用char(1),如果值的長度不固定,有長有短,最好別用char()類型。
二、深入了解mybatis返回null
拋開mybatis框架,回到原始的jdbc查詢,當(dāng)使用oracle的char類型作為條件查詢數(shù)據(jù)時(shí),只有值完全一樣時(shí)才能查到數(shù)據(jù)。
如創(chuàng)建一個(gè)測試表:
create table t_user( user_name char(5) ); insert into t_user (user_name)values('sgl');
select '"'||user_name||'"' from t_user
; -- 查詢結(jié)果為"sgl ",可以看出oracle自動補(bǔ)了兩個(gè)空格
通過jdbc的PreparedStatement方式查詢數(shù)據(jù):
conn=getConnection(); ps=conn.prepareStatement("select * from t_user where user_name=?"); ps.setString(1,"sgl"); ResultSet rs = ps.executeQuery();
通過上面方式是無法查到數(shù)據(jù)的,因?yàn)椴樵儣l件值”sgl”和數(shù)據(jù)庫中值”sgl “是不相等的。
如果值用“sgl ”可以查到數(shù)據(jù):
conn=getConnection(); ps=conn.prepareStatement("select * from t_user where user_name=?"); ps.setString(1,"sgl "); -- 增加兩個(gè)空格不足5位長度 ResultSet rs = ps.executeQuery();
如果使用trim()方式也可以查詢到數(shù)據(jù),如:
conn=getConnection(); ps=conn.prepareStatement("select * from t_user where trim(user_name)=?"); -- 先對數(shù)據(jù)庫中user_name進(jìn)行去空格,然后再比較 ps.setString(1,"sgl"); ResultSet rs = ps.executeQuery();
現(xiàn)在回到mybatis,同事的Mapper文件里查詢sql如下:
main方法內(nèi)容為:
public static void main(String[] args) { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); DataService d = (DataService) ctx.getBean("dataServiceImpl"); Data data = d.selectByName("sgl"); System.out.println(data); }
其實(shí),通過查看源碼或?qū)⑷罩靖臑閐ebug級別,可以看出在mybatis底層,會將查詢語句使用PreparedStatement預(yù)編譯,然后再將參數(shù)設(shè)置進(jìn)去。如下面是mybatis打印出來的日志:
==> Preparing: select * from data where data.name=?
==> Parameters: sgl(String)
根據(jù)前面的jdbc查詢,我們知道原因,所以很容易理解mybatis中的問題。
另外,MySQL下面,當(dāng)char類型字段的值不足時(shí),好像并不自動將值以空格補(bǔ)足,盡管如此,當(dāng)值長度不固定時(shí),也不推薦使用char類型。
jdbc查詢完整的代碼如下:
jdbc工具類:
package com.songguoliang.url; import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.ResultSetMetaData; import java.sql.SQLException; import java.sql.Statement; import java.util.ArrayList; import java.util.List; import java.util.ResourceBundle; /** * 純jdbc連接數(shù)據(jù)類 * @author sgl * */ public class PureJdbcDao { private static ResourceBundle bundle = ResourceBundle.getBundle("jdbc"); private static int reCount = 0; /** * 獲取連接 * @return */ private static Connection getConnection(){ Connection conn=null; try { Class.forName(bundle.getString("driverClassName")); conn = DriverManager.getConnection(bundle.getString("url") , bundle.getString("username") , bundle.getString("password")); } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (SQLException e) { e.printStackTrace(); }finally{ if(null==conn&&reCount<5){ try { Thread.sleep(10000); } catch (InterruptedException e) { e.printStackTrace(); } reCount++; System.out.println("數(shù)據(jù)庫第"+reCount+"次重連"); conn = getConnection(); } } return conn; } /** * 查詢數(shù)據(jù) * @param sql * @return */ public static Listquery(String sql){ List result=new ArrayList (); Connection conn=null; Statement stmt=null; try { //System.out.println("[PureJdbcDao]查詢語句:" + sql); conn=getConnection(); stmt = conn.createStatement(); ResultSet rs = stmt.executeQuery(sql); ResultSetMetaData rsMeta = rs.getMetaData(); while(rs.next()){ int columnNum=rsMeta.getColumnCount(); String []field=new String[columnNum]; String fieldValue=null; for(int i=1;i<=columnNum;i++){ fieldValue=rs.getString(i); if(fieldValue==null){ fieldValue=""; } field[i-1]=fieldValue; } result.add(field); } } catch (SQLException e) { e.printStackTrace(); }finally{ try { if(stmt!=null){ stmt.close(); } if(conn!=null){ conn.close(); } } catch (SQLException e) { e.printStackTrace(); } } return result; } public static List query(String sql,List params){ List result=new ArrayList (); Connection conn=null; PreparedStatement ps=null; try { conn=getConnection(); ps=conn.prepareStatement(sql); for(int i=0;i
測試類:
package com.songguoliang; import java.util.Arrays; import java.util.List; import com.songguoliang.url.PureJdbcDao; public class Test { public static void main(String[] args) { //Listlist=PureJdbcDao.query("select * from t_user where user_name=?",Arrays.asList("sgl")); // 查詢到條數(shù):0 //List list=PureJdbcDao.query("select * from t_user where user_name=?",Arrays.asList("sgl ")); //查詢到條數(shù):1 List list=PureJdbcDao.query("select * from t_user where trim(user_name)=?",Arrays.asList("sgl")); //查詢到條數(shù):1 System.out.println("查詢到條數(shù):"+list.size()); } }
總結(jié)
以上所述是小編給大家介紹的解決mybatis使用char類型字段查詢oracle數(shù)據(jù)庫時(shí)結(jié)果返回null問題,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時(shí)回復(fù)大家的。在此也非常感謝大家對創(chuàng)新互聯(lián)網(wǎng)站的支持!