[轉帖]MySQL的sum函式返回的型別
阿新 • • 發佈:2018-11-08
原帖地址: http://bylijinnan.iteye.com/blog/1984472?utm_source=tuicool
摘要 :
- The SUM() and AVG() functions return a DECIMAL value for exact-value arguments (integer or DECIMAL), and a DOUBLE value for
原文:
今天專案切換資料庫時,出錯
訪問資料庫的程式碼大概是這樣:
- String sql = "select sum(number) as sumNumberOfOneDay from tableName";
- List<Map> rows = getJdbcTemplate().queryForList(sql);
- for (Map row : rows) {
- SomeBean item = new SomeBean();
- item.setSumNumberOfOneDay(objectToInt(row.get("sumNumberOfOneDay")));
- }
- private int objectToInt(Object obj) {
- return Integer.parseInt("" + obj);
- }
表字段“number”的型別是int(10) unsigned
連線資料庫DataBaseA,測試執行正常;切換到另一資料庫DataBaseB(資料庫表,表名,表結構一樣)時,發現報錯:
java.lang.NumberFormatException: For input string: "10.0"
把sql語句拷貝到MySQL命令列窗口裡面直接執行,sum(number)返回的值是10;
但在Spring的getJdbcTemplate().queryForList(sql)返回,則變成了10.0,
列印row.get("sumNumberOfOneDay").getClass()的結果是:class java.lang.Double
切回DataBaseA,列印結果是java.math.BigDecimal
兩個資料庫的查詢結果在MySQL命令列視窗返回整數,但在Java程式中返回浮點數
那不用Spring,直接操作JDBC:
Java程式碼
- Connection conn = getJdbcTemplate().getDataSource().getConnection();
- Statement st = conn.createStatement();
- ResultSet rs = st.executeQuery(sql);
- ResultSetMetaData rsmd = rs.getMetaData();
- for (int i = 1; i <= rsmd.getColumnCount(); i++) {
- String name = rsmd.getColumnName(i);
- String type = rsmd.getColumnTypeName(i);
- System.out.println(name + ", " + type);
- }
DataBaseA列印的結果:
sumNumberOfOneDay, DECIMAL
DataBaseB列印的結果:
sumNumberOfOneDay, DOUBLE
基本可判斷是MySQL的問題
網上搜索一下,果然:
Java程式碼
- The SUM() and AVG() functions return a DECIMAL value for exact-value arguments (integer or DECIMAL), and a DOUBLE value for approximate-value arguments (FLOAT or DOUBLE). (Before MySQL 5.0.3, SUM() and AVG() return DOUBLE for all numeric arguments.)
http://stackoverflow.com/questions/10592481/what-is-the-return-type-of-sum-in-mysql
原來MySQL 5.0.3之前的版本,sum函式返回的是DOUBLE型別
回頭檢查一下MySQL的版本:
DataBaseA:
5.1.44 Source distribution
DataBaseB:
4.1.7-standard-log
果然是這樣
解決辦法:
1.笨方法,就是重寫objectToInt方法:
Java程式碼
- if (obj instanceof Double) {
- return ((Double)obj).intValue();
- }
- if (obj instanceof BigDecimal) {
- return ((BigDecimal)obj).intValue();
- }
- return Integer.parseInt(obj.toString());
2.利用Spring的BeanPropertyRowMapper:
Java程式碼
- List<SomeBean> list = getJdbcTemplate().query(sql, new BeanPropertyRowMapper(SomeBean.class));
應該儘量採用方法2,避免自己處理
檢視一下Spring的getJdbcTemplate().query(sql, new BeanPropertyRowMapper(SomeBean.class))方法,
發現它的大體思路是這樣:
1.通過SomeBean.class得到所有property
2.根據property的型別,呼叫ResultSet.getXXX()得到對應的值
在ResultSet.getXXX()方法裡面,就實現了型別轉換
例如com.mysql.jdbc.Result的getInt方法(關鍵部分的程式碼):
Java程式碼
- val = getString(columnIndex);
- if ((val != null) && (val.length() != 0)) {
- if ((val.indexOf("e") == -1) && (val.indexOf("E") == -1)
- && (val.indexOf(".") == -1)) {
- return Integer.parseInt(val);
- } else {
- // Convert floating point
- return (int) (Double.parseDouble(val));
- }
- } else {
- return 0;
- }