Hibernate5.1+Sqlserver2000分頁查詢
前幾天改到一個bug:從MS SQLserver上面同步表結構並且採集資料寫入其他庫。然後用的核心技術是用的Hibernate。
其中bug出在SQLServer2000版本上。排查下來發現2000版本真的是一個讓人頭疼的資料庫。
驅動jar包不相容;hibernate5.1分頁查詢也不能用。系統表也與其他版本的天差地別。
0|1一、驅動問題
一開始上網查詢,發現大家都推薦用JTDS驅動。但是JTDS貌似不能與官方的Hibernate相容,需要使用第三方Hibernate。
不然Hibernate在建立連線時會丟擲驅動不能轉換的異常。因為要做其他版本相容(程式碼不做大改動),
所以沒換成jtds的驅動(net.sourceforge.jtds.jdbc.Driver)。
然後用了ms2000的三個驅動。測試透過。但是要注意區分驅動和資料庫連線資訊的寫法
1 jdbc.drivers=com.microsoft.jdbc.sqlserver.SQLServerDriver2 jdbc.url=jdbc:microsoft:sqlserver://localhost:1433;DatabaseName=test
在查詢分頁問題時,偶然發現了一個更方便的方法。
資料連結:https://blog.csdn.net/hexin373/article/details/8260752
maven中央庫裡的sqljdbc4是不行的。這裡我特地下載了 sqljdbc_3.0.1301.101_chs.tar.gz。
連結: 提取碼: kwaj
用了這個jar以後,就可以把上面三個ms2000的jar包刪掉了,而且驅動和資料庫連線資訊也可以和其他版本做統一。所以推薦這個方法
1 jdbc.drivers=com.microsoft.sqlserver.jdbc.SQLServerDriver2 jdbc.url=jdbc:sqlserver://localhost:1433;DatabaseName=COREJAVA
0|1二、系統表問題
ms2005版本以後系統表做了很多增刪,比如:sys.extended_properties等。這些在ms2000都沒有。
但是這個問題比較好改。
這邊放出兩段查詢表結構的語句,不過語句還沒來得及最佳化。
private static final String STRUCT_SQL_FORMAT = Joiner.on("n").join(Arrays.asList( "SELECT convert(varchar(100), h.TABLE_CATALOG) AS TABLE_CATALOG, ", "upper(convert(varchar(100), h.TABLE_NAME)) AS TABLE_NAME, ", "convert(varchar(100), h.TABLE_TYPE) AS TABLE_TYPE, ", "convert(varchar(100), h.value ) AS TABLE_COMMENT, ", "upper(convert(varchar(100), a.name)) AS COLUMN_NAME,", "convert(varchar(100), b.name) AS COLUMN_TYPE,", "convert(varchar(100), COLUMNPROPERTY(a.id,a.name,'PRECISION')) AS COLUMN_LENGTH,", "convert(varchar(100), isnull(COLUMNPROPERTY(a.id,a.name,'Scale'),0)) AS NUM_SCALE,", "convert(varchar(100), case when exists(SELECT 1 FROM sysobjects where xtype='PK' and name in (SELECT name FROM sysindexes WHERE indid in( SELECT indid FROM sysindexkeys WHERE id = a.id AND colid=a.colid ))) then 'YES' else 'NO' end) AS IS_PRIMARYKEY, ", "convert(varchar(100), case when a.isnullable=1 then 'YES'else 'NO' end) AS IS_NULLABLE, ", "convert(varchar(100), isnull(g.[value],'')) AS COLUMN_COMMENT, ", "dc.definition COLUMN_DEFAULT", "FROM syscolumns a ", "left join systypes b on a.xusertype = b.xusertype ", "inner join sysobjects d on a.id=d.id and d.xtype='U' and d.name <> 'dtproperties' ", "left join sys.extended_properties g on a.id=g.major_id and a.colid=g.minor_id ", "left join (", " select a.TABLE_CATALOG, a.TABLE_NAME, a.TABLE_TYPE, b.value from information_schema.tables a ", " left join sys.extended_properties b on b.major_id = OBJECT_ID(a.TABLE_NAME) and b.minor_id = 0 ", ") h on h.TABLE_NAME = d.name ", "LEFT JOIN sys.default_constraints dc ON d.id=dc.parent_object_id AND a.colid=dc.parent_column_id AND a.cdefault=dc.[object_id]", "order by d.name " )); private static final String STRUCT_SQL_FORMAT_FOR_2000 = Joiner.on("n").join(Arrays.asList( "select convert(varchar(100), h.TABLE_CATALOG) AS TABLE_CATALOG, ", "upper(convert(varchar(100), h.TABLE_NAME)) AS TABLE_NAME, ", "convert(varchar(100), h.TABLE_TYPE) AS TABLE_TYPE, ", "convert(varchar(100), ta.TABLE_COMMENT ) AS TABLE_COMMENT, ", "upper(convert(varchar(100), ta.NAME)) AS COLUMN_NAME, ", "convert(varchar(100), ta.COLUMN_TYPE) AS COLUMN_TYPE, ", "convert(varchar(100), ta.COLUMN_LENGTH) AS COLUMN_LENGTH, ", "convert(varchar(100), isnull(ta.NUM_SCALE,0)) AS NUM_SCALE, ", "convert(varchar(100), ta.IS_PRIMARYKEY) AS IS_PRIMARYKEY, ", "convert(varchar(100), ta.IS_NULLABLE) AS IS_NULLABLE, ", "convert(varchar(100), ta.COLUMN_COMMENT) AS COLUMN_COMMENT, ", "convert(varchar(100), ta.COLUMN_DEFAULT) AS COLUMN_DEFAULT ", " FROM ", "(SELECT ", " TABLE_NAME = d.name, ", " TABLE_COMMENT = isnull(f. VALUE, ''), ", " NAME = a.name, ", " IS_PRIMARYKEY = CASE WHEN EXISTS (SELECT 1 FROM sysobjects WHERE xtype = 'PK' AND parent_obj = a.id AND name IN ( SELECT name FROM sysindexes WHERE indid IN ( SELECT indid FROM sysindexkeys WHERE id = a.id AND colid = a.colid))) THEN 'YES' ELSE 'NO' END, ", " COLUMN_TYPE = b.name, ", " COLUMN_LENGTH = COLUMNPROPERTY(a.id, a.name, 'PRECISION'), ", " NUM_SCALE = isnull(COLUMNPROPERTY(a.id, a.name, 'Scale'), 0), ", " IS_NULLABLE = CASE WHEN a.isnullable = 1 THEN 'YES' ELSE 'NO' END, ", " COLUMN_DEFAULT = isnull(e. TEXT, ''), ", " COLUMN_COMMENT = isnull(g.[value], '') ", "FROM ", "tsyscolumns a ", "LEFT JOIN systypes b ON a.xusertype = b.xusertype ", "INNER JOIN sysobjects d ON a.id = d.id AND d.xtype = 'U' AND d.name <> 'dtproperties' ", "LEFT JOIN syscomments e ON a.cdefault = e.id ", "LEFT JOIN sysproperties g ON a.id = g.id AND a.colid = g.smallid ", "LEFT JOIN sysproperties f ON d.id = f.id AND f.smallid = 0 ", ") ta ", "left join information_schema.tables h on h.TABLE_NAME = ta.table_name" ));
0|1三、Hibernate5.1分頁問題
最頭疼的問題。我不懂是Hibernate官方根本沒測試過ms2000的分頁,還是我的用法有問題。
1 public <T> List<T> executeQuery(final String sqlString, Integer current, Integer maxResult, Integer fetchSize, Class<T> t, Object... parameters) throws SqlExecutionException { 2 List<T> rowsList; 3 Session session = sessionLocal.get(); 4 try { 5 Query query = session.createSQLQuery(sqlString); 6 if (current != null) { 7 query.setFirstResult(current); 8 } 9 10 if (maxResult != null && maxResult > 0) {11 query = query.setMaxResults(maxResult);12 }13 14 if (fetchSize != null && fetchSize > 0) {15 query = query.setFetchSize(fetchSize);16 }17 18 if (parameters != null && parameters.length > 0) {19 for (int i = 0; i < parameters.length; i++) {20 query.setParameter(i, parameters[i]);21 }22 }23 24 rowsList = query.setResultTransformer(Transformers.aliasToBean(t)).list();25 } catch (Exception e) {26 throw new SqlExecutionException(sqlString, e.getCause());27 } finally {28 session.close();29 session = null;30 sessionLocal.remove();31 }32 return rowsList;33 }
上面程式碼是我做的Hibernate分頁封裝。maxResult實際上相當於pageSize。如果maxResult為空則執行正常。但一旦指定了maxResult,就會報錯。
呼叫上面方法:
1 List<HashMap> data = session.executeQuery(2 "select * from test",3 0,4 100,5 100,6 HashMap.class,7 null8 );
則會丟擲異常:
1 com.syher.hibernate.jdbc.exception.SqlExecutionException: 第 1 行: '@P0' 附近有語法錯誤。
什麼原因導致的?明明就一條簡單的查詢語句啊?
跟蹤Hibernate原始碼到Loader的executeQueryStatement方法。
1 protected SqlStatementWrapper executeQueryStatement( 2 String sqlStatement, 3 QueryParameters queryParameters, 4 boolean scroll, 5 List<AfterLoadAction> afterLoadActions, 6 SessionImplementor session) throws SQLException { 7 8 // Processing query filters. 9 queryParameters.processFilters( sqlStatement, session );10 11 // Applying LIMIT clause.12 final LimitHandler limitHandler = getLimitHandler(13 queryParameters.getRowSelection()14 );15 String sql = limitHandler.processSql( queryParameters.getFilteredSQL(), queryParameters.getRowSelection() );16 17 // Adding locks and comments.18 sql = preprocessSQL( sql, queryParameters, getFactory().getDialect(), afterLoadActions );19 20 final PreparedStatement st = prepareQueryStatement( sql, queryParameters, limitHandler, scroll, session );21 return new SqlStatementWrapper(22 st, getResultSet(23 st,24 queryParameters.getRowSelection(),25 limitHandler,26 queryParameters.hasAutoDiscoverScalarTypes(),27 session28 )29 );30 }
LimitHandler類是把我們的sql語句加工成分頁語句的類。
在這裡,我們的sql語句select * from test 經過limitHandler.processSql方法處理後, 會變成 select top ? * from test;
1 @Override 2 public String processSql(String sql, RowSelection selection) { 3 if (LimitHelper.hasFirstRow( selection )) { 4 throw new UnsupportedOperationException( "query result offset is not supported" ); 5 } 6 7 final int selectIndex = sql.toLowerCase(Locale.ROOT).indexOf( "select" ); 8 final int selectDistinctIndex = sql.toLowerCase(Locale.ROOT).indexOf( "select distinct" ); 9 final int insertionPoint = selectIndex + (selectDistinctIndex == selectIndex ? 15 : 6);10 11 return new StringBuilder( sql.length() + 8 )12 .append( sql )13 .insert( insertionPoint, " TOP ? " )14 .toString();15 }
上網查了一下,jdbc prepareStatement預編譯不支援top ?的寫法。然後我特地寫了個jdbc的demo驗證,發現問題也確實出在jdbc。
@Test public void run() { try { String sql = "SELECT TOP ?* FROM test"; Connection conn = getJDBCConnection(); PreparedStatement pst = conn.prepareStatement(sql); pst.setInt(1, 10); ResultSet rs = pst.executeQuery(); } catch (Exception e) { e.printStackTrace(); } } public static Connection getJDBCConnection() throws IOException { Connection conn = null; String drivers = "com.microsoft.sqlserver.jdbc.SQLServerDriver"; //String drivers = "com.microsoft.jdbc.sqlserver.SQLServerDriver"; String url = "jdbc:sqlserver://192.168.2.173:1433;DatabaseName=dbtest"; //String url = "jdbc:microsoft:sqlserver://192.168.2.173:1433;DatabaseName=dbtest"; //String url = "jdbc:sqlserver://192.168.3.104:1433;DatabaseName=testdb"; String userName = "sa"; String password = "sa@173"; //String password = "sa@104"; if (drivers != null) { try { Class.forName(drivers).newInstance(); conn = DriverManager.getConnection(url, userName, password); } catch (Exception e) { e.printStackTrace(); System.out.println("資料庫連線失敗"); } } else { System.out.println("資料庫驅動不存在"); } return conn; }
上面demo裡,ms2008、ms2016都不會有問題。而只有ms2000丟擲了“@P0' 附近有語法錯誤。”的異常。
抓耳撓腮了兩天之後,終於在一篇部落格裡面找到了靈感。
參考資料:https://gcy6164.iteye.com/blog/1160119
一開始看到這篇資料時,我是確實按著部落格步驟改,結果沒用。遂放棄。
然後某天在TopLimitHandler類中看到了部落格中的supportsLimit方法,才恍然大悟。原來是自己改錯了方向。
我沒看過Hibernate3.2的原始碼,但是大致猜測Hibernate3.2中應該是沒有LimitHandler類的。所以Hibernate3.2中判斷
資料庫是否支援分頁是在SQLServerDialect中。而Hibernate5.1時為了更好的擴充套件,增加了LimitHandler專門處理分頁語句的介面。
而判斷資料庫是否支援分頁的方法也轉移到了這個類中。
因此Hibernate3.2的修改教程不適合Hibernate5.1。但其實是同一個解決思路。
於是我自定義了一個Hibernate方言繼承了SQLServerDialect,並重寫了getLimitHandler方法。
1 public class SQLServer2000Dialect extends SQLServerDialect { 2 3 public SQLServer2000Dialect() { 4 super(); 5 } 6 7 @Override 8 public LimitHandler getLimitHandler() { 9 return new SQLServer2000LimitHandler(false, false);10 }11 }
自定義了SQLServer2000LimitHandler類,並修改了supportsLimit方法。
1 public class SQLServer2000LimitHandler extends TopLimitHandler { 2 public SQLServer2000LimitHandler(boolean supportsVariableLimit, boolean bindLimitParametersFirst) { 3 super(supportsVariableLimit, bindLimitParametersFirst); 4 } 5 6 @Override 7 public boolean supportsLimit() { 8 return false; 9 }10 }
打包,除錯。果然沒問題了。
我的hibernate測試程式碼:
__EOF__
作 者:
出 處:https://www.cnblogs.com/braska/p/10483002.html
版權宣告:本文版權歸作者和部落格園共有,歡迎轉載,但未經作者同意必須保留此段宣告,且在文章頁面明顯位置給出原文連線,否則保留追究法律責任的權利。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/3209/viewspace-2822037/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Elasticsearch 分頁查詢Elasticsearch
- NET 集合分頁查詢
- AntDesignBlazor示例——分頁查詢Blazor
- MySQL的分頁查詢MySql
- ThinkPhp框架:分頁查詢PHP框架
- 分頁查詢優化優化
- elasticsearch查詢之大資料集分頁查詢Elasticsearch大資料
- MySQL分頁查詢優化MySql優化
- indexdb實現分頁查詢Index
- 分庫分表後的分頁查詢
- MySQL——優化巢狀查詢和分頁查詢MySql優化巢狀
- Oracle總結【SQL細節、多表查詢、分組查詢、分頁】OracleSQL
- SSH框架下的分頁查詢框架
- 菜品條件分頁查詢
- 【記錄】SSH分頁查詢功能
- 資料庫全表查詢之-分頁查詢優化資料庫優化
- SSM框架實現分頁查詢例子SSM框架
- MySQL分優化之超大頁查詢MySql優化
- c# winform 實現分頁查詢C#ORM
- (MySQL學習筆記)分頁查詢MySql筆記
- ElasticSearch - 分頁查詢方式二 【scroll】滾動查詢(kibana、Java示例)ElasticsearchJava
- 小書MybatisPlus第4篇-表格分頁與下拉分頁查詢MyBatis
- 關於 groupBy 分組查詢的分頁處理
- 使用Mybatis-plus進行分頁查詢,沒有分頁效果,查詢的資料量超出每頁數量設定MyBatis
- 如何優雅地實現分頁查詢
- HBase學習之二: hbase分頁查詢
- 分頁查詢及其擴充應用案例
- Oracle資料庫中的分頁查詢Oracle資料庫
- SpringMVC+Spring Data JPA +Bootstrap 分頁實現和模糊查詢分頁SpringMVCboot
- 百億級資料分表後怎麼分頁查詢?
- 流式查詢1. mybatis的遊標Cursor,分頁大資料查詢MyBatis大資料
- 查詢——二分查詢
- 百億級資料 分庫分表 後怎麼分頁查詢?
- SSH整合實現分頁查詢(兩種方式)
- OData武裝你的WEBAPI-分頁查詢WebAPI
- elasticsearch查詢之大資料集分頁效能分析Elasticsearch大資料
- 億萬級分庫分表後如何進行跨表分頁查詢
- Yii2:ArrayDataProvider 小改動後支援分頁查詢IDE