- 浏览: 243229 次
- 性别:
- 来自: 广州
文章分类
- 全部博客 (172)
- J2SE学习模块 (35)
- Oracle学习模块 (12)
- Jsp学习模块 (11)
- Servlet学习模块 (1)
- Tomcat 模块 (4)
- Struts1.x学习模块 (5)
- Spring学习模块 (2)
- Hibernate学习模块 (11)
- XML学习模块 (1)
- UML学习模块 (0)
- 算法学习模块 (6)
- 设计模式模块 (2)
- Mysql学习模块 (1)
- SQL_Server学习模块 (8)
- 项目开发模块 (10)
- 搜索引擎 (14)
- 开发工具的使用 (2)
- 面试题集 (7)
- 开发工具 (8)
- Linux (5)
- JavaFX模块 (1)
- 程序与人生 (4)
- 计算机网络 (6)
- EJB学习模块 (1)
- javascript常用模块 (2)
- 英语学习 (1)
- 程序变量命名的几条法则:匈牙利命名法 (1)
- 驼峰式大小写 (1)
- 帕斯卡命名法 (1)
- Jquery控制图片宽度及高度 (1)
- 喜讯--FireFox7.01已经支持CSS中 text-overflow ellipsis 属性 (1)
- 遍历 Map 的三种 常用方法 java (1)
- JDK自带工具Jstat (1)
- 提高 Oracle 查询 效率 建议 (1)
- 常用工具 (1)
最新评论
-
a465492689:
挺好,谢谢分享
存储过程 -
huangqinghe:
ding 顶~~~
Intellij Idea12 中文乱码问题总结 -
Redpick13:
楼主好人啊,有耐心
Java中的二维数组的定义与学习 -
dandongsoft:
神鼎飞丹砂
java.lang.NoClassDefFoundError: org/apache/lucene/index/memory/MemoryIndex -
devil__lord:
不错讲得很清楚 color=#cff
Java中的二维数组的定义与学习
简介: JDBC,即 Java™ Database Connectivity 是 JDK 中最常用的包之一,但是只有极少数开发人员能够充分使用其完整的 — 或最新的 — 功能。Ted Neward 提供了像 ResultSet 这样的最新 JDBC 功能,可以在系统繁忙时自动滚动和更新,无论是否有开放数据库连接,Rowset 都能正常工作,而且批量更新可围绕网络快速执行多条 SQL 语句。
目前,许多开发人员把 Java Database Connectivity (JDBC) API 当作一种数据访问平台,比如 Hibernate 或 SpringMany。然而 JDBC 在数据库连接中不仅仅充当后台角色。对于 JDBC,您了解的越多,您的 RDBMS 交互效率就越高。
在本期 5 件事 系列 中,我将向您介绍几种 JDBC 2.0 到 JDBC 4.0 中新引入的功能。设计时考虑到现代软件开发所面临的挑战,这些新特性支持应用程序可伸缩性,并提高开发人员的工作效率 — 这是现代 Java 开发人员面临的两个最常见的挑战。
1. 标量函数
不同的 RDBMS 实现对 SQL 和/或增值特性(目的是让程序员的工作更为简单)提供不规则的支持。例如,众所周知,SQL 支持一个标量运算 COUNT(),返回满足特定 SQL 过滤规则的行数(更确切地说,是 WHERE 谓词)。除此之外,修改 SQL 返回的值是很棘手的 — 想要从数据库获取当前日期和时间会使 JDBC 开发人员、甚至最有耐心的程序员发疯(甚至是心力憔悴)。
于是,JDBC 规范针对不同的 RDBMS 实现通过标量函数提供一定程度的隔离/改写。JDBC 规范包括一系列受支持的操作,JDBC 驱动程序应该根据特定数据库实现的需要进行识别和改写。因此,对于一个支持返回当前日期和/或时间的数据库,时间查询应当如清单 1 那样简单:
清单 1. 当前时间?
JDBC API 识别的标量函数完整列表在 JDBC 规范附录中给出(见 参考资料),但是给定的驱动程序或数据库可能不支持完整列表。您可以使用从 Connection 返回的 DatabaseMetaData 对象来获取给定 JDBC 支持的函数,如清单 2 所示:
清单 2. 能为我提供什么?
标量函数列表是从各种 DatabaseMetaData 方法返回的一个逗号分隔的 String。例如,所有数值标量由 getNumericFunctions() 调用列出,在结果上执行一个 String.split() — 瞧! — 即刻出现 equals()-testable 列表。
--------------------------------------------------------------------------------
2. 可滚动 ResultSets
创建一个 Connection 对象,并用它来创建一个 Statement,这在 JDBC 中是最常用的。提供给 SQL SELECT 的 Statement 返回一个 ResultSet。然后,通过一个 while 循环(和 Iterator 没什么不同)得到 ResultSet,直到 ResultSet 为空,循环体从左到右的每次提取一列。
这整个操作过程是如此普遍,近乎神圣:它这样做只是因为它应该这样做。唉!实际上这是完全没必要的。
引入可滚动 ResultSet
许多开发人员没有意识到,在过去的几年中 JBDC 已经有了相当大的增强,尽管这些增强在新版本中已经有所反映。 第一次重大增强是在 JDBC 2.0 中,发生在使用 JDK 1.2 期间。写这篇文章时,JDBC 已经发展到了 JDBC 4.0。
JDBC 2.0 中一个有趣的增强(尽管常常被忽略)是 ResultSet 的滚动功能,这意味着您可以根据需要前进或者后退,或者两者均可。这样做需要一点前瞻性,然而 — JDBC 调用必须指出在创建 Statement 时需要一个可以滚动的 ResultSet。
验证 ResultSet 类型如果您怀疑一个驱动程序事实上可能不支持可滚动的 ResultSets,不管 DatabaseMetaData 中是如何写的,您都要调用 getType() 来验证 ResultSet 类型。当然,如果您是个偏执的人,您可能也不相信 getType() 的返回值。可以这样说,如果 getType() 隐瞒关于 ResultSet 的返回值,它们确实是 要吃定您。
如果底层 JDBC 驱动程序支持滚动,一个可滚动的 ResultSet 将从那个 Statement 返回。但是在请求它之前最好弄清楚驱动程序是否支持可滚动性。您可以通过 DatabaseMetaData 对象探询滚动性,如上所述,这个对象可从任何 Connection 中获取。
一旦您有了一个 DatabaseMetaData 对象,一个对 getJDBCMajorVersion() 的调用将会确定驱动程序是否支持 JDBC 规范,至少是 JDBC 2.0 规范。当然一个驱动程序可能会隐瞒它对给定规范的支持程度,因此为了安全起见,用期望得到的 ResultSet 类型调用 supportsResultSetType() 方法。(在 ResultSet 类上它是一个常量;稍后我们将对其每个值进行讨论。)
清单 3. 可以滚动?
请求一个可滚动的 ResultSet
假设您的驱动程序回答 “是”(如果不是,您需要一个新的驱动程序或数据库),您可以通过传递两个参数到 Connection.createStatement() 调用来请求一个可滚动的 ResultSet,如清单 4 所示:
清单 4. 我想要滚动!
在调用 createStatement() 时,您必须特别小心,因为它的第一个和第二个参数都是 int 的。(在 Java 5 之前我们不能使用枚举类型!)任何 int 值(包括错误的常量)对 createStatement() 都有效。
第一个参数,指定 ResultSet 中期望得到的 “可滚动性”,应该是以下 3 个值之一:
ResultSet.TYPE_FORWARD_ONLY:这是默认的,是我们了解且喜欢的流水游标。
ResultSet.TYPE_SCROLL_INSENSITIVE:这个 ResultSet 支持向后迭代以及向前迭代,但是,如果数据库中的数据发生变化,ResultSet 将不能反映出来。这个可滚动的 ResultSet 可能是最常用到的类型。
ResultSet.TYPE_SCROLL_SENSITIVE:所创建的 ResultSet 不但支持双向迭代,而且当数据库中的数据发生变化时还为您提供一个 “实时” 数据视图。
第二个参数在下一个技巧中介绍,稍等片刻。
定向滚动
当您从 Statement 获取一个 ResultSet 后,通过它向后滚动只需调用 previous(),即向后滚动一行,而不是向前,就像 next() 那样。您也可以调用 first() 返回到 ResultSet 开头,或者调用 last() 转到 ResultSet 的末尾,或者...您自己拿主意。
relative() 和 absolute() 方法也是很有用的:前者移动指定数量的行(如果是正数则向前移动,是负数则向后移动),后者移动 ResultSet 中指定数量的行,不管游标在哪。当然,目前行数是由 getRow() 获取的。
如果您打算通过调用 setFetchDirection() 在一个特定方向进行一些滚动,可以通过指定方向来帮助 ResultSet。(无论向哪个方向滚动,对于 ResultSet 都可行,但是预先知道滚动方向可以优化其数据检索。)
--------------------------------------------------------------------------------
3. 可更新的 ResultSets
JDBC 不仅仅支持双向 ResultSet,也支持就地更新 ResultSet。这就是说,与其创建一个新 SQL 语句来修改目前存储在数据库中的值,您只需要修改保存在 ResultSet 中的值,之后该值会被自动发送到数据库中该行所对应的列。
请求一个可更新的 ResultSet 类似于请求一个可滚动的 ResultSet 的过程。事实上,在此您将为 createStatement() 使用第二个参数。您不需要为第二个参数指定 ResultSet.CONCUR_READ_ONLY,只需要发送 ResultSet.CONCUR_UPDATEABLE 即可,如清单 5 所示:
清单 5. 我想要一个可更新的 ResultSet
假设您的驱动程序支持可更新光标(这是 JDBC 2.0 规范的另一个特性,这是大多数 “现实” 数据库所支持的),您可以更新 ResultSet 中任何给定的值,方法是导航到该行并调用它的一个 update...() 方法(如清单 6 所示),如同 ResultSet 的 get...()方法。在 ResultSet 中 update...() 对于实际的列类型是超负荷的。因此要更改名为 “PRICE” 的浮点列,调用 updateFloat("PRICE")。然而,这样做只能更新 ResultSet 中的值。为了将该值插入支持它的数据库中,可以调用 updateRow()。如果用户改变调整价格的想法,调用 cancelRowUpdates() 可以停止所有正在进行的更新。
清单 6. 一个更好的方法
JDBC 2.0 不只支持更新。如果用户想要添加一个全新的行,不需要创建一个新 Statement 并执行一个 INSERT,只需要调用 moveToInsertRow(),为每个列调用 update...(),然后调用 insertRow() 完成工作。如果没有指定一个列值,数据库会默认将其看作 SQL NULL(如果数据库模式不允许该列为 NULL,这可能触发 SQLException)。
当然,如果 ResultSet 支持更新一行,也必然支持通过 deleteRow() 删除一行。
差点忘了强调一点,所有这些可滚动性和可更新性都适用于 PreparedStatement(通过向 prepareStatement() 方法传递参数),由于一直处于 SQL 注入攻击的危险中,这比一个规则的 Statement 好很多。
--------------------------------------------------------------------------------
4. Rowsets
既然所有这些功能在 JDBC 中大约有 10 年了,为什么大多数开发人员仍然迷恋向前滚动 ResultSet 和不连贯访问?
罪魁祸首是可伸缩性。保持最低的数据库连接是支持大量用户通过 Internet 访问公司网站的关键。因为滚动和/或更新 ResultSet 通常需要一个开放的网络连接,而许多开发人员通常不(或不能)使用这些连接。
幸好,JDBC 3.0 引入另一种解决方案让您同样可以做很多之前使用 ResultSet 方可以做的事情,而不需要数据库连接保持开放状态。
从概念上讲,Rowset 本质上是一个 ResultSet,但是它支持连接模型或断开模型,您所需要做的是创建一个 Rowset,将其指向一个 ResultSet,当它完成自我填充之后,将其作为一个 ResultSet,如清单 7 所示:
清单 7. Rowset 取代 ResultSet
JDBC 还附带了 5 个 Rowset 接口 “实现”(也就是扩展接口)。JdbcRowSet 是一个连接的 Rowset 实现;其余 4 个是断开的:
CachedRowSet 只是一个断开的 Rowset.
WebRowSet 是 CachedRowSet 的一个子集,知道如何将其结果转换成 XML,并再次转换回来。
JoinRowSet 是一个 WebRowSet,知道如何形成一个 SQL JOIN,而无需连接到数据库。
FilteredRowSet 是一个 WebRowSet,知道如何更进一步过滤传递回来的数据,而不需要连接到数据库。
Rowsets 是完整的 JavaBeans,意味着它们支持侦听类事件,因此,如果需要,也可以捕捉、检查并执行对 Rowset 的任何修改。事实上,如果 Rowset 有自己的 Username、Password、URL 和 DatasourceName 属性集(这意味着它将使用 DriverManager.getConnection() 创建一个连接)或者 Datasource 属性集(这很可能由 JNDI 获取),它甚至能管理对数据库的全部操作。然后,您可以在 Command 属性中指定要执行的 SQL,调用 execute(),然后处理结果 — 不需要更多的工作。
通常,Rowset 实现是由 JDBC 驱动程序提供的,因此实际的名称和/或包由您所使用的 JDBC 驱动程序决定。从 Java 5 开始 Rowset 实现已经是标准版本(standard distribution)的一部分了,因此您只需要创建一个 ...RowsetImpl(),然后让其运行。(如果您的驱动程序不能提供一个参考实现,Sun 提供了一个,参见 参考资料 部分的链接。)
--------------------------------------------------------------------------------
5. 批量更新
尽管 Rowset 很实用,但有时候也不能满足您的需求,您可能需要返回来直接编写 SQL 语句。在这种情况下,特别是当您面对一大堆工作时,您就会很感激批量更新功能,可在一个网络往返行程中在数据库中执行多条 SQL 语句。
要确定 JDBC 驱动程序是否支持批量更新,快速调用 DatabaseMetaData.supportsBatchUpdates() 可产生一个明示支持与否的布尔值。在支持批量更新时(由一些非 SELECT 标示),所有任务逐个排队然后在某一瞬间同时得到更新,如清单 8 所示:
清单 8. 让数据库进行批量更新!
默认必须调用 setAutoCommit(),驱动程序会试图交付提供给它的每条语句。除此之外,其余代码都是简单易懂的:使用 Statement 或 PreparedStatement 进行常见 SQL 操作,但是不调用 execute(),而调用 executeBatch(),排队等候调用而不是立即发送。
准备好各种语句之后,在数据库中使用 executeBatch() 触发所有的语句,这将返回一组整型值,每个值保存同样的结果,好像使用了 executeUpdate() 一样。
在批量处理的一条语句发生错误的情况下,如果驱动程序不支持批量更新,或者批处理中的一条语句返回 ResultSet,驱动程序将抛出一个 BatchUpdateException。有时候,在抛出一个异常之后,驱动程序可能试着继续执行语句。JDBC 规范不能授权某一行为,因此您应该事先试用驱动程序,这样就可以确切地知道它是如何工作的。(当然,您要执行单元测试,确保在错误成为问题之前发现它,对吧?)
--------------------------------------------------------------------------------
结束语
作为 Java 开发的一个主题,JDBC API 是每个开发人员应该熟知的,就像您的左右手那样。有趣的是,在过去的几年中,许多开发人员并不了解 API 的增强功能,因此,他们错失了本文所讲到的省时技巧。
当然,您是否决定使用 JDBC 的新功能取决于您自己。需要考虑的一个关键因素是您所使用的系统的可伸缩性。对伸缩性的要求越高,对数据库的使用就越受限制,因此而减少的网络流量就会越多。Rowset、标量调用和批量更新将会是给您带来帮助的益友。另外,尝试可滚动和可更新的 ResultSet(这不像 Rowset 那样耗内存),并度量可伸缩性。这可能没您想象的糟糕。
转载自:http://www.ibm.com/developerworks/cn/java/j-5things10.html
目前,许多开发人员把 Java Database Connectivity (JDBC) API 当作一种数据访问平台,比如 Hibernate 或 SpringMany。然而 JDBC 在数据库连接中不仅仅充当后台角色。对于 JDBC,您了解的越多,您的 RDBMS 交互效率就越高。
在本期 5 件事 系列 中,我将向您介绍几种 JDBC 2.0 到 JDBC 4.0 中新引入的功能。设计时考虑到现代软件开发所面临的挑战,这些新特性支持应用程序可伸缩性,并提高开发人员的工作效率 — 这是现代 Java 开发人员面临的两个最常见的挑战。
1. 标量函数
不同的 RDBMS 实现对 SQL 和/或增值特性(目的是让程序员的工作更为简单)提供不规则的支持。例如,众所周知,SQL 支持一个标量运算 COUNT(),返回满足特定 SQL 过滤规则的行数(更确切地说,是 WHERE 谓词)。除此之外,修改 SQL 返回的值是很棘手的 — 想要从数据库获取当前日期和时间会使 JDBC 开发人员、甚至最有耐心的程序员发疯(甚至是心力憔悴)。
于是,JDBC 规范针对不同的 RDBMS 实现通过标量函数提供一定程度的隔离/改写。JDBC 规范包括一系列受支持的操作,JDBC 驱动程序应该根据特定数据库实现的需要进行识别和改写。因此,对于一个支持返回当前日期和/或时间的数据库,时间查询应当如清单 1 那样简单:
清单 1. 当前时间?
Connection conn = ...; // get it from someplace Statement stmt = conn.createStatement(); ResultSet rs = stmt.executeQuery("{CURRENT_DATE()}");
JDBC API 识别的标量函数完整列表在 JDBC 规范附录中给出(见 参考资料),但是给定的驱动程序或数据库可能不支持完整列表。您可以使用从 Connection 返回的 DatabaseMetaData 对象来获取给定 JDBC 支持的函数,如清单 2 所示:
清单 2. 能为我提供什么?
Connection conn = ...; // get it from someplace DatabaseMetaData dbmd = conn.getMetaData();
标量函数列表是从各种 DatabaseMetaData 方法返回的一个逗号分隔的 String。例如,所有数值标量由 getNumericFunctions() 调用列出,在结果上执行一个 String.split() — 瞧! — 即刻出现 equals()-testable 列表。
--------------------------------------------------------------------------------
2. 可滚动 ResultSets
创建一个 Connection 对象,并用它来创建一个 Statement,这在 JDBC 中是最常用的。提供给 SQL SELECT 的 Statement 返回一个 ResultSet。然后,通过一个 while 循环(和 Iterator 没什么不同)得到 ResultSet,直到 ResultSet 为空,循环体从左到右的每次提取一列。
这整个操作过程是如此普遍,近乎神圣:它这样做只是因为它应该这样做。唉!实际上这是完全没必要的。
引入可滚动 ResultSet
许多开发人员没有意识到,在过去的几年中 JBDC 已经有了相当大的增强,尽管这些增强在新版本中已经有所反映。 第一次重大增强是在 JDBC 2.0 中,发生在使用 JDK 1.2 期间。写这篇文章时,JDBC 已经发展到了 JDBC 4.0。
JDBC 2.0 中一个有趣的增强(尽管常常被忽略)是 ResultSet 的滚动功能,这意味着您可以根据需要前进或者后退,或者两者均可。这样做需要一点前瞻性,然而 — JDBC 调用必须指出在创建 Statement 时需要一个可以滚动的 ResultSet。
验证 ResultSet 类型如果您怀疑一个驱动程序事实上可能不支持可滚动的 ResultSets,不管 DatabaseMetaData 中是如何写的,您都要调用 getType() 来验证 ResultSet 类型。当然,如果您是个偏执的人,您可能也不相信 getType() 的返回值。可以这样说,如果 getType() 隐瞒关于 ResultSet 的返回值,它们确实是 要吃定您。
如果底层 JDBC 驱动程序支持滚动,一个可滚动的 ResultSet 将从那个 Statement 返回。但是在请求它之前最好弄清楚驱动程序是否支持可滚动性。您可以通过 DatabaseMetaData 对象探询滚动性,如上所述,这个对象可从任何 Connection 中获取。
一旦您有了一个 DatabaseMetaData 对象,一个对 getJDBCMajorVersion() 的调用将会确定驱动程序是否支持 JDBC 规范,至少是 JDBC 2.0 规范。当然一个驱动程序可能会隐瞒它对给定规范的支持程度,因此为了安全起见,用期望得到的 ResultSet 类型调用 supportsResultSetType() 方法。(在 ResultSet 类上它是一个常量;稍后我们将对其每个值进行讨论。)
清单 3. 可以滚动?
int JDBCVersion = dbmd.getJDBCMajorVersion(); boolean srs = dbmd.supportsResultSetType(ResultSet.TYPE_SCROLL_INSENSITIVE); if (JDBCVersion > 2 || srs == true) { // scroll, baby, scroll! }
请求一个可滚动的 ResultSet
假设您的驱动程序回答 “是”(如果不是,您需要一个新的驱动程序或数据库),您可以通过传递两个参数到 Connection.createStatement() 调用来请求一个可滚动的 ResultSet,如清单 4 所示:
清单 4. 我想要滚动!
Statement stmt = con.createStatement( ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); ResultSet scrollingRS = stmt.executeQuery("SELECT * FROM whatever");
在调用 createStatement() 时,您必须特别小心,因为它的第一个和第二个参数都是 int 的。(在 Java 5 之前我们不能使用枚举类型!)任何 int 值(包括错误的常量)对 createStatement() 都有效。
第一个参数,指定 ResultSet 中期望得到的 “可滚动性”,应该是以下 3 个值之一:
ResultSet.TYPE_FORWARD_ONLY:这是默认的,是我们了解且喜欢的流水游标。
ResultSet.TYPE_SCROLL_INSENSITIVE:这个 ResultSet 支持向后迭代以及向前迭代,但是,如果数据库中的数据发生变化,ResultSet 将不能反映出来。这个可滚动的 ResultSet 可能是最常用到的类型。
ResultSet.TYPE_SCROLL_SENSITIVE:所创建的 ResultSet 不但支持双向迭代,而且当数据库中的数据发生变化时还为您提供一个 “实时” 数据视图。
第二个参数在下一个技巧中介绍,稍等片刻。
定向滚动
当您从 Statement 获取一个 ResultSet 后,通过它向后滚动只需调用 previous(),即向后滚动一行,而不是向前,就像 next() 那样。您也可以调用 first() 返回到 ResultSet 开头,或者调用 last() 转到 ResultSet 的末尾,或者...您自己拿主意。
relative() 和 absolute() 方法也是很有用的:前者移动指定数量的行(如果是正数则向前移动,是负数则向后移动),后者移动 ResultSet 中指定数量的行,不管游标在哪。当然,目前行数是由 getRow() 获取的。
如果您打算通过调用 setFetchDirection() 在一个特定方向进行一些滚动,可以通过指定方向来帮助 ResultSet。(无论向哪个方向滚动,对于 ResultSet 都可行,但是预先知道滚动方向可以优化其数据检索。)
--------------------------------------------------------------------------------
3. 可更新的 ResultSets
JDBC 不仅仅支持双向 ResultSet,也支持就地更新 ResultSet。这就是说,与其创建一个新 SQL 语句来修改目前存储在数据库中的值,您只需要修改保存在 ResultSet 中的值,之后该值会被自动发送到数据库中该行所对应的列。
请求一个可更新的 ResultSet 类似于请求一个可滚动的 ResultSet 的过程。事实上,在此您将为 createStatement() 使用第二个参数。您不需要为第二个参数指定 ResultSet.CONCUR_READ_ONLY,只需要发送 ResultSet.CONCUR_UPDATEABLE 即可,如清单 5 所示:
清单 5. 我想要一个可更新的 ResultSet
Statement stmt = con.createStatement( ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_UPDATEABLE); ResultSet scrollingRS = stmt.executeQuery("SELECT * FROM whatever");
假设您的驱动程序支持可更新光标(这是 JDBC 2.0 规范的另一个特性,这是大多数 “现实” 数据库所支持的),您可以更新 ResultSet 中任何给定的值,方法是导航到该行并调用它的一个 update...() 方法(如清单 6 所示),如同 ResultSet 的 get...()方法。在 ResultSet 中 update...() 对于实际的列类型是超负荷的。因此要更改名为 “PRICE” 的浮点列,调用 updateFloat("PRICE")。然而,这样做只能更新 ResultSet 中的值。为了将该值插入支持它的数据库中,可以调用 updateRow()。如果用户改变调整价格的想法,调用 cancelRowUpdates() 可以停止所有正在进行的更新。
清单 6. 一个更好的方法
Statement stmt = con.createStatement( ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_UPDATEABLE); ResultSet scrollingRS = stmt.executeQuery("SELECT * FROM lineitem WHERE id=1"); scrollingRS.first(); scrollingRS.udpateFloat("PRICE", 121.45f); // ... if (userSaidOK) scrollingRS.updateRow(); else scrollingRS.cancelRowUpdates();
JDBC 2.0 不只支持更新。如果用户想要添加一个全新的行,不需要创建一个新 Statement 并执行一个 INSERT,只需要调用 moveToInsertRow(),为每个列调用 update...(),然后调用 insertRow() 完成工作。如果没有指定一个列值,数据库会默认将其看作 SQL NULL(如果数据库模式不允许该列为 NULL,这可能触发 SQLException)。
当然,如果 ResultSet 支持更新一行,也必然支持通过 deleteRow() 删除一行。
差点忘了强调一点,所有这些可滚动性和可更新性都适用于 PreparedStatement(通过向 prepareStatement() 方法传递参数),由于一直处于 SQL 注入攻击的危险中,这比一个规则的 Statement 好很多。
--------------------------------------------------------------------------------
4. Rowsets
既然所有这些功能在 JDBC 中大约有 10 年了,为什么大多数开发人员仍然迷恋向前滚动 ResultSet 和不连贯访问?
罪魁祸首是可伸缩性。保持最低的数据库连接是支持大量用户通过 Internet 访问公司网站的关键。因为滚动和/或更新 ResultSet 通常需要一个开放的网络连接,而许多开发人员通常不(或不能)使用这些连接。
幸好,JDBC 3.0 引入另一种解决方案让您同样可以做很多之前使用 ResultSet 方可以做的事情,而不需要数据库连接保持开放状态。
从概念上讲,Rowset 本质上是一个 ResultSet,但是它支持连接模型或断开模型,您所需要做的是创建一个 Rowset,将其指向一个 ResultSet,当它完成自我填充之后,将其作为一个 ResultSet,如清单 7 所示:
清单 7. Rowset 取代 ResultSet
Statement stmt = con.createStatement( ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_UPDATEABLE); ResultSet scrollingRS = stmt.executeQuery("SELECT * FROM whatever"); if (wantsConnected) JdbcRowSet rs = new JdbcRowSet(scrollingRS); // connected else CachedRowSet crs = new CachedRowSet(scrollingRS); disconnected
JDBC 还附带了 5 个 Rowset 接口 “实现”(也就是扩展接口)。JdbcRowSet 是一个连接的 Rowset 实现;其余 4 个是断开的:
CachedRowSet 只是一个断开的 Rowset.
WebRowSet 是 CachedRowSet 的一个子集,知道如何将其结果转换成 XML,并再次转换回来。
JoinRowSet 是一个 WebRowSet,知道如何形成一个 SQL JOIN,而无需连接到数据库。
FilteredRowSet 是一个 WebRowSet,知道如何更进一步过滤传递回来的数据,而不需要连接到数据库。
Rowsets 是完整的 JavaBeans,意味着它们支持侦听类事件,因此,如果需要,也可以捕捉、检查并执行对 Rowset 的任何修改。事实上,如果 Rowset 有自己的 Username、Password、URL 和 DatasourceName 属性集(这意味着它将使用 DriverManager.getConnection() 创建一个连接)或者 Datasource 属性集(这很可能由 JNDI 获取),它甚至能管理对数据库的全部操作。然后,您可以在 Command 属性中指定要执行的 SQL,调用 execute(),然后处理结果 — 不需要更多的工作。
通常,Rowset 实现是由 JDBC 驱动程序提供的,因此实际的名称和/或包由您所使用的 JDBC 驱动程序决定。从 Java 5 开始 Rowset 实现已经是标准版本(standard distribution)的一部分了,因此您只需要创建一个 ...RowsetImpl(),然后让其运行。(如果您的驱动程序不能提供一个参考实现,Sun 提供了一个,参见 参考资料 部分的链接。)
--------------------------------------------------------------------------------
5. 批量更新
尽管 Rowset 很实用,但有时候也不能满足您的需求,您可能需要返回来直接编写 SQL 语句。在这种情况下,特别是当您面对一大堆工作时,您就会很感激批量更新功能,可在一个网络往返行程中在数据库中执行多条 SQL 语句。
要确定 JDBC 驱动程序是否支持批量更新,快速调用 DatabaseMetaData.supportsBatchUpdates() 可产生一个明示支持与否的布尔值。在支持批量更新时(由一些非 SELECT 标示),所有任务逐个排队然后在某一瞬间同时得到更新,如清单 8 所示:
清单 8. 让数据库进行批量更新!
conn.setAutoCommit(false); PreparedStatement pstmt = conn.prepareStatement("INSERT INTO lineitems VALUES(?,?,?,?)"); pstmt.setInt(1, 1); pstmt.setString(2, "52919-49278"); pstmt.setFloat(3, 49.99); pstmt.setBoolean(4, true); pstmt.addBatch(); // rinse, lather, repeat int[] updateCount = pstmt.executeBatch(); conn.commit(); conn.setAutoCommit(true);
默认必须调用 setAutoCommit(),驱动程序会试图交付提供给它的每条语句。除此之外,其余代码都是简单易懂的:使用 Statement 或 PreparedStatement 进行常见 SQL 操作,但是不调用 execute(),而调用 executeBatch(),排队等候调用而不是立即发送。
准备好各种语句之后,在数据库中使用 executeBatch() 触发所有的语句,这将返回一组整型值,每个值保存同样的结果,好像使用了 executeUpdate() 一样。
在批量处理的一条语句发生错误的情况下,如果驱动程序不支持批量更新,或者批处理中的一条语句返回 ResultSet,驱动程序将抛出一个 BatchUpdateException。有时候,在抛出一个异常之后,驱动程序可能试着继续执行语句。JDBC 规范不能授权某一行为,因此您应该事先试用驱动程序,这样就可以确切地知道它是如何工作的。(当然,您要执行单元测试,确保在错误成为问题之前发现它,对吧?)
--------------------------------------------------------------------------------
结束语
作为 Java 开发的一个主题,JDBC API 是每个开发人员应该熟知的,就像您的左右手那样。有趣的是,在过去的几年中,许多开发人员并不了解 API 的增强功能,因此,他们错失了本文所讲到的省时技巧。
当然,您是否决定使用 JDBC 的新功能取决于您自己。需要考虑的一个关键因素是您所使用的系统的可伸缩性。对伸缩性的要求越高,对数据库的使用就越受限制,因此而减少的网络流量就会越多。Rowset、标量调用和批量更新将会是给您带来帮助的益友。另外,尝试可滚动和可更新的 ResultSet(这不像 Rowset 那样耗内存),并度量可伸缩性。这可能没您想象的糟糕。
转载自:http://www.ibm.com/developerworks/cn/java/j-5things10.html
发表评论
-
巧用JDK自带工具Jstat查看Gc垃圾回收之使用方法
2012-07-19 09:40 3929如何查看JVM中内存使用情况如何? JVM垃圾回收是否正常? ... -
遍历Map的三种常用方法 java
2011-11-09 12:06 28911、一般的map定义: Map<Key, Object& ... -
程序变量命名的几条法则:匈牙利命名法,驼峰式大小写,帕斯卡命名法
2011-09-07 17:54 51421.匈牙利命名法的好处: ... -
JDK源码中对当前时间的一些用法
2011-09-05 14:51 1089JDK源码中对当前时间的一些用法: String[] ... -
Eclipse安装反编译工具查看Jar包源码 JD Eclipse插件
2011-08-24 10:37 5099如何在Eclipse中安装反编译工具查看Jar包源码呢? 打 ... -
Struts2图片上传成功后,异常:java.lang.IllegalStateException
2011-07-10 18:49 19581.场景: 使用Struts2图片上传成功后出现异常: ... -
J2EE表生成器@TableGenerator
2011-03-31 14:29 1740最近工作中遇到了许多 ... -
Spring Hibernate Extjs开发指南
2011-03-23 18:08 989从学习到找工作,然后再面试,最后才经历上班; 从这段时间对自 ... -
获取当前时间的前三个月 java
2011-03-17 11:14 4762获取当前时间的前三个月: Date date = n ... -
java 线程的状态与控制
2010-12-16 20:54 1131一、线程的状态 线程的主要状态有: 1.新 ... -
MyEclipse Blue 6.5 注册码 可到2013年12月
2010-12-06 09:44 1527做Java软件开发的,首选的软件工具莫非是MyEclip ... -
关于 Java Collections API 您不知道的 5 件事,第 2 部分
2010-10-24 00:28 529简介: 您可以在任何地方使用 Java™ 集合,但是一定要小心 ... -
关于 Java 常用工具您不知道的 5 件事
2010-10-24 00:25 532简介: 有些 Java™ 工具无法分类,只能算作 “有用的东西 ... -
关于 JVM 命令行标志您不知道的 5 件事
2010-10-24 00:22 774简介: Java™ 虚拟机有数百个命令行选项,被经验丰富的开发 ... -
关于 Java Scripting API 您不知道的 5 件事
2010-10-24 00:17 589简介: Java™ 语言足以满足您的一些项目的需求,但是脚本语 ... -
关于 Java 性能监控您不知道的 5 件事,第 2 部分
2010-10-24 00:15 570简介: 如果 JDK 中配置有全功能分析器 JConsole ... -
关于 Java 性能监控您不知道的 5 件事,第 1 部分
2010-10-24 00:13 770简介: 责怪糟糕的代码 ... -
关于 JAR 您不知道的 5 件事
2010-10-24 00:11 412简介: 除了一些基础的 JAR 之外,许多 Java™ 开发人 ... -
关于 java.util.concurrent 您不知道的 5 件事,第 2 部分
2010-10-24 00:05 487简介: 除了具有很好的并发性的 Collections,jav ... -
关于 java.util.concurrent 您不知道的 5 件事,第 1 部分
2010-10-24 00:02 496简介: 编写能够良好执行,防止应用程序受损的多线程代码是很艰巨 ...
相关推荐
JDBC(Java DataBase Connectivity,java数据库连接)是一种用于执行SQL语句的Java API,可以为多种关系数据库提供统一访问,它由一组用Java语言编写的类和接口组成。JDBC提供了一种基准,据此可以构建更高级的工具和...
然而,对于特定数据库系统的连接,可能需要相应的ODBC(Open Database Connectivity)或JDBC(Java Database Connectivity)驱动程序。 通过使用Labview 2018 Database Connectivity Toolkit(32位),开发者可以...
了解JDBC (Java Database Connectivity)相关概念 JDBC的四种类型的驱动 JDBC核心接口与类 JDBC体系结构图和开发步骤 JDBC 事务 进阶目标: 并发控制(Concurrency Control) 事务的隔离级别 批处理(Batch...
基于Java使用JDBC(Java Database Connectivity)操作MySQL数据库的简单项目示例
此外,该工具包还支持ODBC(Open Database Connectivity)和JDBC(Java Database Connectivity),使得连接范围进一步扩大到任何兼容这些标准的数据库系统。 在V2.7.0.346版本中,可能包含了以下改进和修复: 1. ...
LabVIEW提供了数据库连接库,使得通过ODBC(Open Database Connectivity)或JDBC(Java Database Connectivity)接口与多种类型的数据库(如MySQL、Oracle、SQL Server等)建立连接变得简单。 2. **创建表格**:`2...
### 跨界融合:Matlab与Java Database Connectivity(JDBC)的协同之旅 #### Matlab概述 Matlab是一款高性能的数值计算和可视化软件环境,由MathWorks公司开发。它被广泛应用于工程计算、算法开发、数据分析、可视...
《Java Database Programming with JDBC》这本书深入探讨了Java数据库编程的核心技术——JDBC(Java Database Connectivity)。JDBC是Java平台的标准接口,它允许Java程序与各种类型的数据库进行交互,包括关系型...
这个"Java_Database_Connectivity_JDBC_2012.rar_oracle"压缩包文件,特别是结合其描述“oracle分布式数据库java连接服务器指导”,显然是关于如何使用Java通过JDBC连接到Oracle分布式数据库的详细教程或指南。...
1. **连接数据库**: 使用JDBC(Java Database Connectivity)接口建立与HSQL数据库的连接。通过`Class.forName()`加载驱动,然后使用`DriverManager.getConnection()`创建连接。 2. **创建表**: 通过SQL语句创建表,...
文章主要是JDBC的两种架构:BS架构(Browser and server)浏览器服务器架构-------》web、javaweb、phpweb CS架构(Client and server)客户端服务器架构---------》QQ、微信 主要解析BS架构: ...
《数据库编程与JDBC和JAVA》是一本专为IT专业人士准备的指南,旨在帮助读者深入理解和熟练运用JDBC(Java Database Connectivity)进行数据库编程。本书适用于熟悉Java编程基础,但对数据库和JDBC技术不熟悉的开发者...
在Java中,与关系型数据库如Access交互通常通过JDBC(Java Database Connectivity)实现。JDBC是Java平台的标准部分,提供了一组接口和类,允许开发者连接到各种类型的数据库,包括Access。要访问.Nwind.mdb文件,...
在Java中,通过JDBC(Java Database Connectivity)接口可以调用SQL语句。 2. **JDBC**:JDBC是Java平台的一个标准API,它允许Java程序与各种数据库进行通信。通过JDBC,我们可以执行SQL查询、事务处理以及数据库...
在Java中,我们可以使用JDBC(Java Database Connectivity)API来实现SQL语句的执行。JDBC为Java程序员提供了一个统一的接口,可以用来连接各种不同类型的数据库,包括MySQL、Oracle、SQL Server等。 "access_java_...
- Java Database Connectivity (JDBC) 是Java平台的标准API,用于与各种类型的数据库进行交互。在这些示例中,开发者可能使用JDBC驱动程序连接到数据库,执行SQL语句,如查询、插入、更新和删除数据。 2. **...
Java提供了一种标准的方式来与各种数据库进行交互,即使用Java Database Connectivity (JDBC) API。JDBC API允许Java程序通过Java代码来执行SQL语句,实现数据的增删查改。本资源包含的几个jar包正是用于连接不同...
1. **JDBC(Java Database Connectivity)**: JDBC是Java平台的标准接口,允许Java应用程序与各种类型的数据库进行通信。它由一组Java类和接口组成,如`java.sql.DriverManager`、`Connection`、`Statement`和`...
在Java的技术生态中,JDBC(Java Database Connectivity)允许程序员使用统一的接口连接各种关系型数据库;EJB(Enterprise JavaBeans)简化了跨平台企业级应用的开发、部署和管理;RMI(Remote Method Invocation)...
这个“Java_Database_CreateTable_module.rar”压缩包文件显然包含了一个关于如何在Java中创建数据库表的教程或代码示例。在这个模块中,我们将深入探讨Java数据库编程的基础知识,特别是如何使用SQL语句在数据库中...