(8) 使用函数 尽管大部分关系数据库系统( R D B M S)都扩充了可以在S Q L中使用的函数,许多数据库系 统还允许用户自己扩充函数,但下面的几个函数总是可以使用的: AVG 返回某一组中的值除以该组中值的个数的和。 COUNT 返回一组行或值中行或值的个数。 MAX 返回一组值中的最大值。 MIN 返回一组值中的最小值。 下面是实际的例子: 求出所有订单的数量总和: SELECT COUNT(Freight) FROM orders 求出所有订单的运费平均值: SELECT AVG(Freight) FROM orders 求出所有订单的运费最大值 SELECT MAX(Freight) FROM orders 求出所有订单的运费最小值 SELECT MIN(Freight) FROM orders (9) 子查询 第二章预备知识59
子查询的概念在于将一个查询的结果作为另一个查询的条件,举例如下: 在订单数据表中的客户公司是使用公司代号来表示的,如果需要查询运费在5 0 0以上的公司 的名称和电话,就需要使用子查询这个概念: SELECT CompanyName,phone FROM customers WHERE CustomerID IN ( SELECT CustomerID from orders WHERE freight>500 ) 结果得到了需要的数据: C o m p a n y N a m e p h o n e --------------------- -------------------------- Ernst Handel 7 6 7 5 - 3 4 2 5 Great Lakes Food Market (503) 555-7555 Hungry Owl All-Night Grocers 2967 542 Queen Cozinha ( 11) 555-11 8 9 Q U I C K - S t o p 0 3 7 2 - 0 3 5 1 8 8 Rattlesnake Canyon Grocery (505) 555-5939 Save-a-lot Markets (208) 555-8097 White Clover Markets (206) 555-411 2 2. 使用数据修改命令 S Q L语言中数据的修改命令包括: I N S E RT 建立记录。 D E L E T E 删除记录。 U P D AT E 修改记录。 (1) INSERT语句 I N S E RT语句在使用时有两种不同的格式。需要注意的是, I N S E RT语句假定需要插入数据 的数据表已经用C R E AT E语句或其他工具建立。 第一种用法是不列出数据表的各个列名,而按照数据表建立时的顺序将数据列出: INSERT INTO Customers VALUES ('AAAAA','AAAAA Co.Ltd.','riso liao','Owner','Peking University','Bei Jing','Bei Jing','HaiDian','100871','86-10-62754178','86-10-62763126') 第二种用法是在数据表的后面按照后面数据需要插入的列的顺序列出数据表中各个列的名称: INSERT INTO Customers( CustomerID,CompanyName,ContactName,ContactTitle,Address,City,Region,PostalCode, Country,Phone,Fax ) VALUES ('AAAAA','AAAAA Co.Ltd.','riso liao','Owner','Peking University','Bei Jing','Bei Jing','HaiDian','100871','86-10-62754178','86-10-62763126') 60第一部分JSP 入门
上面两条语句的作用都一样,不过在实际使用中建议使用第二种方法,因为第二种方法可 以让数据和数据要插入的列一一对应,而且还有利于插入空值,例如:现在需要加入到记录中 的这个公司数据相当不完整,只有公司名称和代号,当采用第一种方法时,需要这样书写S Q L 语句: INSERT INTO Customers VALUES ('AAAAA','AAAAA Co.Ltd.',NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL) 而第二种方法只需要书写如下语句即可: INSERT INTO Customers( CustomerID,CompanyName) VALUES ('AAAAA','AAAAA Co.Ltd.') (2) DELETE语句 D E L E T E语句的使用是相当简单的,具体就是: DELETE FROM 表名条件 其中条件不是必需的,当没有条件时,就意味着删除表中的所有记录。 例如,语句DELETE FROM customers WHERE CustomerID=”A A A A A”将删除刚才建立的 记录;而语句DELETE FROM customers将删除c u s t o m e r s数据表中的所有记录。 (3) UPDAT E语句 U P D AT E语句的作用是将数据库中某一条记录的某一个记录域更新,语句格式如下: UPDATE 数据表SET 列名=新数据条件 和D E L E T E语句一样,这里的条件也可以是没有的,如果没有条件,那么数据表中的每一条 记录都将被更新。 现在来试验一下U P D AT E语句: 首先看一看数据本来的样子: SELECT CustomerID,CompanyName FROM customers 可以看到第一行记录为: C u s t o m e r I D C o m p a n y N a m e ------------------ -------------------------- A L F K I Alfreds Futterkiste 现在执行一个U P D AT E: UPDATE customers SET CompanyName='AAAAA' WHERE CustomerID='ALFKI' 再执行一下SELECT CustomerID,CompanyName FROM customers 发现结果变成了: C u s t o m e r I D CompanyName ---------------- ------------------------ A L F K I A A A A A 第二章预备知识61
这样,数据库中的一条语句就被更新了。 2.5 JDBC 本节将在上节讲述的S Q L语言的基础上介绍J D B C,J D B C使得在J a v a程序中可以轻松地操纵 数据库:从企业级的O r a c l e、S y b a s e、D B 2到最简单的A c c e s s、M y S Q L。在J S P中,就是利用 J D B C来访问数据库的。 2.5.1 什么是J D B C JDBC 是一种用于执行SQL 语句的Java API,它由一组用Java 编程语言编写的类和接口组 成。JDBC 为工具/数据库开发人员提供了一个标准的A P I,使他们能够用纯Java API 来编写数 据库应用程序。 有了J D B C,向各种关系数据库发送SQL 语句就是一件很容易的事。换言之,有了J D B C A P I,就不必为访问Sybase 数据库专门写一个程序,为访问Oracle 数据库又专门写一个程序, 为访问Informix 数据库又写另一个程序,等等。只需用JDBC API 写一个程序就够了,它可向 相应的数据库发送SQL 语句。而且,使用Java 编程语言编写的应用程序,无须去忧虑要为不同 的平台编写不同的应用程序。将Java 和JDBC 结合起来将使程序员只需写一遍程序就可让它在 任何平台上运行。 Java 具有坚固、安全、易于使用、易于理解和可从网络上自动等特性,是编写数据库 应用程序的杰出语言。所需要的只是Java 应用程序与各种不同数据库之间进行对话的方法。而 JDBC 正是作为此种用途的机制。 JDBC 扩展了Java 的功能。例如,用Java 和JDBC API 可以发布含有Applet 的网页,而该 applet 使用的信息可能来自远程数据库。企业也可以用JDBC 通过Intranet 将所有职员连到一个 或多个内部数据库中(即使这些职员所用的计算机有Wi n d o w s、Macintosh 和Unix 等各种不 同的操作系统)。随着越来越多的程序员开始使用Java 编程语言,对从Java 中便捷地访问数据 库的要求也在日益增加。 MIS 管理员们都喜欢Java 和JDBC 的结合,因为它使信息传播变得容易和经济。企业可继 续使用它们安装好的数据库,并能便捷地存取信息,即使这些信息是存储在不同数据库管理系 统上。新程序的开发期很短。安装和版本控制将大为简化。程序员可只编写一遍应用程序或只 更新一次,然后将它放到服务器上,随后任何人就都可得到最新版本的应用程序。对于商务上 的销售信息服务, Java 和JDBC 可为外部客户提供获取信息的更新更好方法。 1. JDBC 的用途 简单地说,JDBC 可做三件事: • 与数据库建立连接。 • 发送SQL 语句。 • 处理结果。 下列代码段给出了以上三步的基本示例: Connection con = DriverManager.getConnection ('jdbc:odbc:wombat', 'login', 62第一部分JSP 入门
'password'); Statement stmt = con.createStatement(); ResultSet rs = stmt.executeQuery('SELECT a, b, c FROM Table1'); while (rs.next()) { int x = rs.getInt('a'); String s = rs.getString('b'); float f = rs.getFloat('c'); } 2. JDBC 是一种低级API ,是高级API 的基础 JDBC 是个”低级”接口,就是说,它用于直接调用SQL 命令。在这方面它的功能极佳, 并比其他的数据库连接A P I更易于使用,但它同时也被设计为一种基础接口,在它之上可以建 立高级接口和工具。高级接口是”对用户友好的”接口,它使用的是一种更易理解和更为方便 的A P I,这种API 在幕后被转换为诸如JDBC 这样的低级接口。在编写本文时,正在开发两种 基于JDBC 的高级A P I: 一种是用于Java 的嵌入式S Q L。至少已经有一个提供者计划编写它。DBMS 实现S Q L:一 种专门设计来与数据库联合使用的语言。JDBC 要求SQL 语句必须作为String 传给Java 方法。 相反,嵌入式SQL 预处理器允许程序员将SQL 语句直接与Java 混在一起使用。例如,可在 SQL 语句中使用Java 变量,用以接受或提供SQL 值。然后,嵌入式SQL 预处理器将通过 JDBC 调用把这种Java/SQL 的混合物转换为J a v a。 另一种关系数据库表到Java 类的直接映射。JavaSoft 和其他提供者都声称要实现该A P I。 在这种”对象/关系”映射中,表中的每行对应于类的一个实例,而每列的值对应于该实例的一 个属性。于是,程序员可直接对Java 对象进行操作;存取数据所需的SQL 调用将在”掩盖下” 自动生成。此外还可提供更复杂的映射,例如将多个表中的行结合进一个Java 类中。 随着人们对JDBC 的兴趣日益浓厚,越来越多的开发人员一直在使用基于JDBC 的工具, 以使程序的编写更加容易。程序员也一直在编写力图使最终用户对数据库的访问变得更为简单 的应用程序。例如,应用程序可提供一个选择数据库任务的菜单。任务被选定后,应用程序将 给出提示及空白供填写执行选定任务所需的信息。所需信息输入后,应用程序将自动调用所需 的SQL 命令。在这样一种程序的协助下,即使用户根本不懂SQL 的语法,也可以执行数据库 任务。 3. JDBC 与ODBC 和其他API 的比较 目前,O D B C(开放式数据库连接) API 可能是使用最广的、用于访问关系数 据库的编程接口。它能在几乎所有平台上连接几乎所有的数据库。为什么Java 不使用 O D B C? 对这个问题的回答是: Java 可以使用O D B C,但最好是在JDBC 的帮助下以J D B C - O D B C 桥的形式使用,这一点稍后再讲解。现在的问题已变成:”为什么需要J D B C”? 回答如下: ODBC 不适合直接在Java 中使用,因为它使用C 语言接口。从Java 调用本地C 代码在安 全性、实现性、坚固性和程序的自动移植性方面都有许多缺点。 从ODBC C API 到Java API 的字面翻译是不可取的。例如, Java 没有指针,而ODBC 却对 指针用得很广泛(包括很容易出错的指针“void *”)。你可以将JDBC 想像成被转换为面向对 第二章预备知识63
象接口的O D B C,而面向对象的接口对Java 程序员来说较易于接收。 ODBC 很难学。它把简单和高级功能混在一起,而且即使对于简单的查询,其选项也极为 复杂。相反, JDBC 尽量保证简单功能的简便性,而同时在必要时允许使用高级功能。启用” 纯Java “机制需要像JDBC 这样的Java API。如果使用O D B C,就必须手工地将ODBC 驱动程 序管理器和驱动程序安装在每台客户机上。如果完全用Java 编写JDBC 驱动程序,则JDBC 代 码在所有Java 平台上(从网络计算机到大型机)都可以自动安装、移植并保证安全性。 总之,JDBC API 对于基本的SQL 抽象和概念是一种自然的Java 接口。它建立在ODBC 上, 而不是从零开始。因此,熟悉ODBC 的程序员将发现JDBC 很容易使用。JDBC 保留了O D B C 的基本设计特征;事实上,两种接口都基于X/Open SQL CLI(调用级接口)。它们之间最大的 区别在于:JDBC 以Java 风格与优点为基础并进行优化,因此更加易于使用。 最近,Microsoft 又引进了ODBC 之外的新A P I: R D O、ADO 和OLE DB。这些设计在许 多方面与JDBC 是相同的,即它们都是面向对象的数据库接口且基于可在ODBC 上实现的类。 但在这些接口中,未看见有特别的功能使我们要转而选择它们来替代O D B C,尤其是在O D B C 驱动程序已建立起较为完善的市场的情况下。它们最多也就是在ODBC 上加了一种装饰而已。 这并不是说JDBC 不需要从其最初的版本再发展了;然而,我们觉得大部份的新功能应归入诸 如前一节中所述的对象/关系映射和嵌入式SQL 这样的高级A P I。 4. 两层模型和三层模型 JDBC API 既支持数据库访问的两层模型,同时也支持三层模型。 在两层模型中,Java applet 或应用程序将直接与数据库进行对话。这将需要一个JDBC 驱动 程序来与所访问的特定数据库管理系统进行通信。用户的SQL 语句被送往数据库中,而其结果 将被送回给用户。数据库可以位于另一台计算机上,用户通过网络连接到上面。这就叫做客户 机/服务器配置,其中用户的计算机为客户机,提供数据库的计算机为服务器。网络可以是 I n t r a n e t(它可将公司职员连接起来),也可以是I n t e r n e t。 在三层模型中,命令先是被发送到服务的”中间层“,然后由它将SQL 语句发送给数据库。 数据库对SQL 语句进行处理并将结果送回到中间层,中间层再将结果送回给用户。MIS 管理员 们都发现三层模型很吸引人,因为可用中间层来控制对公司数据的访问和可作的更新的种类。 中间层的另一个好处是,用户可以利用易于使用的高级A P I,而中间层将把它转换为相应的低 级调用。而且,许多情况下三层结构可提供一些性能上的好处。 到目前为止,中间层通常都用C 或C++ 这类语言来编写,这些语言执行速度较快。然而, 随着最优化编译器(它把Java 字节代码转换为高效的特定于机器的代码)的引入,用Java 来实 现中间层将变得越来越实际。这将是一个很大的进步,它使人们可以充分利用Java 的诸多优点 (如坚固、多线程和安全等特征)。JDBC 对于从Java 的中间层来访问数据库非常重要。 5. SQL 的一致性 结构化查询语言(SQL) 是访问关系数据库的标准语言。其困难之处在于:虽然大多数的 D B M S(数据库管理系统)对其基本功能都使用了标准形式的S Q L,但它们却不符合最近为更 高级的功能定义的标准SQL 语法或语义。例如,并非所有的数据库都支持存储程序或外部连接, 那些支持这一功能的数据库又相互不一致。人们希望SQL 中真正标准的那部份能够进行扩展以
包括越来越多的功能。但同时JDBC API 又必须支持现有的S Q L。 JDBC API 解决这个问题的一种方法是允许将任何查询字符串一直传到所涉及的DBMS 驱 动程序上。这意味着应用程序可以使用任意多的SQL 功能,但它必须冒这样的风险:有可能在 某些DBMS 上出错。事实上,应用程序查询甚至不一定是S Q L,或者说它可以是为特定的 DBMS 设计的SQL 的专用派生物(例如,文档或图像查询)。 JDBC 处理SQL 一致性问题的第二种方法是提供ODBC 风格的转义子句。 转义语法为几个常见的SQL 分歧提供了一种标准的JDBC 语法。例如,对日期文字和已存 储过程的调用都有转义语法。 对于复杂的应用程序, JDBC 用第三种方法来处理SQL 的一致性问题。它利用D a t a b a s e MetaData 接口来提供关于DBMS 的描述性信息,从而使应用程序能适应每个DBMS 的要求和 功能。 由于JDBC API 将用做开发高级数据库访问工具和API 的基础,因此它还必须注意其所有 上层建筑的一致性。“符合JDBC 标准” 代表用户可依赖的JDBC 功能的标准级别。要使用这一 说明,驱动程序至少必须支持ANSI SQL-2 Entry Level(ANSI SQL-2 代表美国国家标准局1 9 9 2 年所采用的标准。Entry Level 代表SQL 功能的特定清单)。驱动程序开发人员可用JDBC API 所带的测试工具包来确定他们的驱动程序是否符合这些标准。 “符合JDBC 标准” 表示提供者的JDBC 实现已经通过了JavaSoft 提供的一致性测试。这些 一致性测试将检查JDBC API 中定义的所有类和方法是否都存在,并尽可能地检查程序是否具 有SQL Entry Level 功能。当然,这些测试并不完全,而且JavaSoft 目前也无意对各提供者的实 现进行标级。但这种一致性定义的确可对JDBC 实现提供一定的可信度。随着越来越多的数据 库提供者、连接提供者、Internet 提供者和应用程序编程员对JDBC API 的接受,JDBC 也正迅 速成为Java 数据库访问的标准。
|