在AS400 / DB2数据库上运行SELECT
查询时遇到一些麻烦。
当我运行以下代码时,我得到一个Exception
,指出光标无效。
String jdbcURL = "jdbc:as400://10.1.2.200";
DriverManager.registerDriver(new com.ibm.as400.access.AS400JDBCDriver());
Properties props = new Properties();
props.setProperty("user", "tracktool");
props.setProperty("password", "tooltrack1");
con = DriverManager.getConnection(jdbcURL, props);
stmt = con.createStatement(ResultSet.TYPE_FORWARD_ONLY, ResultSet.CONCUR_READ_ONLY);
int count = 0;
try {
String sql = "select * from MVXJDTALR.CSYNBR where CNNBTY='ZZ'";
assertTrue("Select lieferte kein ResultSet.", stmt.execute(sql));
assertTrue("Keine Results im Resultset", stmt.getResultSet().first());
count = stmt.getResultSet().getInt("CNNBNR");
assertTrue("ResultSet hatte falsche Anzahl Spalten", count > 0);
} catch (Exception ex) {
ex.printStackTrace();
fail("Konnte den aktuellen Stand von CSYNBR nicht auslesen!");
}
仅更改此行后
stmt = con.createStatement(ResultSet.TYPE_FORWARD_ONLY, ResultSet.CONCUR_READ_ONLY);
要
stmt = con.createStatement(ResultSet.TYPE_SCROLL_SENSITIVE, ResultSet.CONCUR_READ_ONLY);
它完美无缺。 我在互联网上发现,它与使用它时对结果集进行的更新有关。但我没有做任何更新,也不想知道它们。我只是尝试读取一行并希望从该行中获得一个单独的值。我可以直接读取单个值而无需读取整行,但这也不起作用。
发生异常时,这是一个调用跟踪。
在例外之前记录:
Toolbox for Java - Open Source Software, JTOpen 6.0, codebase 5722-JC1 V5R4M0.6
Properties (7481705) : access = "all".
Properties (7481705) : block size = "32".
Properties (7481705) : block criteria = "2".
Properties (7481705) : date format = "".
Properties (7481705) : date separator = "".
Properties (7481705) : decimal separator = "".
Properties (7481705) : errors = "basic".
Properties (7481705) : extended dynamic = "false".
Properties (7481705) : libraries = "".
Properties (7481705) : naming = "sql".
Properties (7481705) : package = "".
Properties (7481705) : package add = "true".
Properties (7481705) : package cache = "false".
Properties (7481705) : package clear = "false".
Properties (7481705) : package error = "warning".
Properties (7481705) : package library = "".
Properties (7481705) : password = "".
Properties (7481705) : prefetch = "true".
Properties (7481705) : prompt = "".
Properties (7481705) : remarks = "system".
Properties (7481705) : sort = "hex".
Properties (7481705) : sort language = "ENU".
Properties (7481705) : sort table = "".
Properties (7481705) : sort weight = "shared".
Properties (7481705) : time format = "".
Properties (7481705) : time separator = "".
Properties (7481705) : trace = "true".
Properties (7481705) : transaction isolation = "read uncommitted".
Properties (7481705) : translate binary = "false".
Properties (7481705) : user = "tracktool".
Properties (7481705) : package criteria = "default".
Properties (7481705) : lob threshold = "32768".
Properties (7481705) : secure = "false".
Properties (7481705) : data truncation = "true".
Properties (7481705) : proxy server = "".
Properties (7481705) : secondary URL = "".
Properties (7481705) : data compression = "true".
Properties (7481705) : big decimal = "true".
Properties (7481705) : thread used = "true".
Properties (7481705) : cursor hold = "true".
Properties (7481705) : lazy close = "false".
Properties (7481705) : driver = "toolbox".
Properties (7481705) : bidi string type = "".
Properties (7481705) : key ring name = "".
Properties (7481705) : key ring password = "".
Properties (7481705) : full open = "false".
Properties (7481705) : server trace = "0".
Properties (7481705) : database name = "".
Properties (7481705) : extended metadata = "false".
Properties (7481705) : cursor sensitivity = "asensitive".
Properties (7481705) : behavior override = "0".
Properties (7481705) : package ccsid = "13488".
Properties (7481705) : minimum divide scale = "0".
Properties (7481705) : maximum precision = "31".
Properties (7481705) : maximum scale = "31".
Properties (7481705) : translate hex = "character".
Properties (7481705) : toolbox trace = "".
Properties (7481705) : qaqqinilib = "".
Properties (7481705) : login timeout = "".
Properties (7481705) : true autocommit = "false".
Properties (7481705) : bidi implicit reordering = "true".
Properties (7481705) : bidi numeric ordering = "false".
Properties (7481705) : hold input locators = "true".
Properties (7481705) : hold statements = "false".
Properties (7481705) : rollback cursor hold = "false".
Properties (7481705) : variable field compression = "true".
Properties (7481705) : query optimize goal = "0".
Properties (7481705) : keep alive = "".
Properties (7481705) : receive buffer size = "".
Properties (7481705) : send buffer size = "".
Properties (7481705) : XA loosely coupled support = "0".
Properties (7481705) : translate boolean = "true".
Properties (7481705) : metadata source = "1".
Properties (7481705) : query storage limit = "-1".
Properties (7481705) : decfloat rounding mode = "half even".
Properties (7481705) : autocommit exception = "false".
Driver AS/400 Toolbox for Java JDBC Driver (21790187) : Using IBM Toolbox for Java JDBC driver implementation.
Toolbox for Java - Open Source Software, JTOpen 6.0, codebase 5722-JC1 V5R4M0.6
JDBC Level: 30
Connection 10.1.2.200 (21576085) : Client CCSID = 13488.
Connection 10.1.2.200 (21576085) : Setting server NLV = 2929.
Connection 10.1.2.200 (21576085) : Client functional level = V5R4M01 .
Connection 10.1.2.200 (21576085) : Data compression = RLE.
Connection 10.1.2.200 (21576085) : ROWID supported = true.
Connection 10.1.2.200 (21576085) : True auto-commit supported = true.
Connection 10.1.2.200 (21576085) : 128 byte column names supported = true.
Connection 10.1.2.200 (21576085) : Maximum decimal precision = 31.
Connection 10.1.2.200 (21576085) : Maximum decimal scale = 31.
Connection 10.1.2.200 (21576085) : Minimum divide scale = 0.
Connection 10.1.2.200 (21576085) : Translate hex = character.
Connection 10.1.2.200 (21576085) : query optimize goal = 0.
Connection 10.1.2.200 (21576085) : query storage limit = -1.
Connection 10.1.2.200 (21576085) : Using extended datastreams.
Connection 10.1.2.200 (21576085) : JDBC driver major version = 7.
Connection 10.1.2.200 (21576085) : i5/OS VRM = V6R1M0.
Connection 10.1.2.200 (21576085) : Server CCSID = 37.
Connection 10.1.2.200 (21576085) : Server functional level = V6R1M00014 (14).
Connection 10.1.2.200 (21576085) : Server job identifier = 546098/QUSER/QZDASOINIT.
Properties (7481705) : decimal separator = ".".
Properties (7481705) : date format = "dmy".
Properties (7481705) : date separator = ".".
Properties (7481705) : time format = "hms".
Properties (7481705) : time separator = ":".
Connection LR59227P (21576085) open.
Connection LR59227P (21576085) : Auto commit = "true".
Connection LR59227P (21576085) : Read only = "false".
Connection LR59227P (21576085) : Transaction isolation = "1".
Statement STMT0001 (16678784) open. Parent: Connection LR59227P (21576085) .
Statement STMT0001 (16678784) : Escape processing = "true".
Statement STMT0001 (16678784) : Fetch direction = "1000".
Statement STMT0001 (16678784) : Fetch size = "0".
Statement STMT0001 (16678784) : Max field size = "0".
Statement STMT0001 (16678784) : Max rows = "0".
Statement STMT0001 (16678784) : Query timeout = "0".
Statement STMT0001 (16678784) : Result set concurrency = "1007".
Statement STMT0001 (16678784) : Result set holdability = "-9999".
Statement STMT0001 (16678784) : Result set type = "1003".
Statement STMT0001 (16678784) : Behavior Override = "0".
Statement STMT0001 (16678784) : Data to correlate statement with cursor Cursor CRSR0001 (6598415) .
Statement STMT0001 (16678784) : Executing SQL Statement -->[select * from MVXJDTALR.CSYNBR where CNNBTY='ZZ'].
Statement STMT0001 (16678784) : Prepared STMT0001*, SQL Statement -->[select * from MVXJDTALR.CSYNBR where CNNBTY='ZZ'].
Cursor CRSR0001 (6598415) open.
Cursor CRSR0001 (6598415) closed.
ResultSet CRSR0001 (20035600) open. Parent: Statement STMT0001 (16678784) .
ResultSet CRSR0001 (20035600) : Conncurrency = "1007".
ResultSet CRSR0001 (20035600) : Fetch direction = "1000".
ResultSet CRSR0001 (20035600) : Fetch size = "0".
ResultSet CRSR0001 (20035600) : Max rows = "0".
ResultSet CRSR0001 (20035600) : Type = "1003".
Statement STMT0001 (16678784) : Executed STMT0001*, SQL Statement --> [select * from MVXJDTALR.CSYNBR where CNNBTY='ZZ'].
Statement STMT0001 (16678784) : Update count = -1.
Statement STMT0001 (16678784) : Result set = true.
Statement STMT0001 (16678784) : Number of result sets = 0.
Statement STMT0001 (16678784) : Row count estimate = 1.
实际例外:
static method: Throwing exception, sqlState: 24000 reason: Cursor state not valid. vendor code -99999.java.sql.SQLException: Cursor state not valid.
at com.ibm.as400.access.JDError.throwSQLException(JDError.java:389)
at com.ibm.as400.access.JDError.throwSQLException(JDError.java:366)
at com.ibm.as400.access.AS400JDBCResultSet.beforePositioning(AS400JDBCResultSet.java:1234)
at com.ibm.as400.access.AS400JDBCResultSet.first(AS400JDBCResultSet.java:1343)
at com.lr.tracktool.barcode.tests.ConnectionTest.testSQLSelectAndUpdateNew(ConnectionTest.java:389)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:601)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:45)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:42)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:263)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:68)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:47)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:231)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:60)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:229)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:50)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:222)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
at org.junit.runners.ParentRunner.run(ParentRunner.java:300)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
java.sql.SQLException: Cursor state not valid.
at com.ibm.as400.access.JDError.throwSQLException(JDError.java:389)
at com.ibm.as400.access.JDError.throwSQLException(JDError.java:366)
at com.ibm.as400.access.AS400JDBCResultSet.beforePositioning(AS400JDBCResultSet.java:1234)
at com.ibm.as400.access.AS400JDBCResultSet.first(AS400JDBCResultSet.java:1343)
at com.lr.tracktool.barcode.tests.ConnectionTest.testSQLSelectAndUpdateNew(ConnectionTest.java:389)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:601)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:45)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:42)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:263)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:68)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:47)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:231)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:60)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:229)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:50)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:222)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
at org.junit.runners.ParentRunner.run(ParentRunner.java:300)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
ResultSet CRSR0001 (20035600) closed.
Statement STMT0001 (16678784) closed.
Connection LR59227P (21576085) closed.
答案 0 :(得分:4)
在TYPE_FORWARD_ONLY
ResultSet
上调用first()
需要通过JDBC规范抛出SQLException
:( 强调我的)
<强>抛出:强>
SQLException
- 如果发生数据库访问错误;在封闭的结果集上调用此方法或结果集类型为TYPE_FORWARD_ONLY
这正是您的代码中发生的事情。如果您想检查ResultSet
是否有一行,请使用next()
。保证next()
适用于所有结果集类型。
答案 1 :(得分:3)
你的发现是正确的。但是,让我清楚解释为什么会发生这种情况。
根据Oracle文档:
这就是为什么当你改变它然后它开始工作的答案。这是因为代码没有使用executeQuery()
而是使用execute()
。这样做的主要原因是,只有在语句可能返回多个ResultSet对象时才应使用execute方法。我在下面提供了两个链接,其中一个是上面两个要点的来源,第二个来源是关于executeQuery()
和execute()
的解释。
答案 2 :(得分:0)
其实我不是从你的长篇文章中了解你的问题。但是,从您的标题我想告诉您TYPE_FORWARD_ONLY
和TYPE_SCROLL_SENSITIVE
之间的区别。
如果选择TYPE_FORWARD_ONLY
,则光标将仅向前移动。您无法向后移动,即您只能在一个轨道中读取任何一行。
如果您选择TYPE_SCROLL_SENSITIVE
,那么您也可以多次反向移动。
如果您想再提问,请总结您的问题。