友情提示:如果本网页打开太慢或显示不完整,请尝试鼠标右键“刷新”本网页!
第三电子书 返回本书目录 加入书签 我的书架 我的书签 TXT全本下载 『收藏到我的浏览器』

Java编程思想第4版[中文版](PDF格式)-第135部分

快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部! 如果本书没有阅读完,想下次继续接着阅读,可使用上方 "收藏到我的浏览器" 功能 和 "加入书签" 功能!


   

  

  

  

填好这个表单并提交出去以后,会得到一个简单的文本页,其中包含了解析出来的结果。从中可知道CGI 程 

序是否在正常工作。  

当然,用一个程序片来提交数据显得更有趣一些。然而,POST 数据的提交属于一个不同的过程。在用常规方 

式调用了CGI 程序以后,必须另行建立与服务器的一个连接,以便将查询字串反馈给它。服务器随后会进行 

一番处理,再通过标准输入将查询字串反馈回 CGI 程序。  

为建立与服务器的一个直接连接,必须取得自己创建的URL,然后调用 openConnection()创建一个 

URLConnection 。但是,由于URLConnection 一般不允许我们把数据发给它,所以必须很可笑地调用 

setDoOutput(true)函数,同时调用的还包括setDoInput(true)以及setAllowUserInteraction(false)—— 

注释⑥。最后,可调用 getOutputStream()来创建一个 OutputStream (输出数据流),并把它封装到一个 

DataOutputStream 里,以便能按传统方式同它通信。下面列出的便是一个用于完成上述工作的程序片,必须 

在从它的各个字段里收集了数据之后再执行它:  

  

//: POSTtest。java  

// An applet that sends its data via a CGI POST  

import java。awt。*;  

import java。applet。*;  

import java。*;  



                                                                                         574 


…………………………………………………………Page 576……………………………………………………………

import java。io。*;  

  

public class POSTtest extends Applet {  

  final static int SIZE = 10;  

  Button submit = new Button(〃Submit〃);  

  TextField'' t = new TextField'SIZE';  

  String query = 〃〃;  

  Label l = new Label();  

  TextArea ta = new TextArea(15; 60);  

  public void init() {  

    Panel p = new Panel();  

    p。setLayout(new GridLayout(t。length + 2; 2));  

    for(int i = 0; i 《 t。length; i++) {  

      p。add(new Label(  

        〃Field 〃 + i + 〃  〃; Label。RIGHT));  

      p。add(t'i' = new TextField(30));  

    }  

    p。add(l);  

    p。add(submit);  

    add(〃North〃; p);  

    add(〃South〃; ta);  

  }  

  public boolean action (Event evt; Object arg) {  

    if(evt。target。equals(submit)) {  

      query = 〃〃;  

      ta。setText(〃〃);  

      // Encode the query from the field data:  

      for(int i = 0; i 《 t。length; i++)  

         query += 〃Field〃 + i + 〃=〃 +   

           URLEncoder。encode(  

             t'i'。getText()。trim()) +  

           〃&〃;  

      query += 〃submit=Submit〃;  

      // Send the name using CGI's POST process:  

      try {  

        URL u = new URL(  

          getDocumentBase(); 〃cgi…bin/POSTtest〃);  

        URLConnection urlc = u。openConnection();  

        urlc。setDoOutput(true);  

        urlc。setDoInput(true);  

        urlc。setAllowUserInteraction(false);  

        DataOutputStream server =   

          new DataOutputStream(  

            urlc。getOutputStream());  

        // Send the data  

        server。writeBytes(query);  

        server。close();  

        // Read and display the response。 You  

        // cannot use   

        // getAppletContext()。showDocument(u);  

        // to display the results as a Web page!  

        DataInputStream in =   



                                                                                          575 


…………………………………………………………Page 577……………………………………………………………

          new DataInputStream(  

            urlc。getInputStream());  

        String s;  

        while((s = in。readLine()) != null) {  

          ta。appendText(s + 〃n〃);  

        }  

        in。close();  

      }  

      catch (Exception e) {  

        l。setText(e。toString());  

      }  

    }  

    else return super。action(evt; arg);  

    return true;  

  }  

} ///:~  

  

⑥:我不得不说自己并没有真正理解这儿都发生了什么事情,这些概念都是从Elliotte Rusty Harold 编著 

的《Java Network Programming》里得来的,该书由O'Reilly 于 1997 年出版。他在书中提到了Java 连网函 

数库中出现的许多令人迷惑的Bug。所以一旦涉足这些领域,事情就不是编写代码,然后让它自己运行那么 

简单。一定要警惕潜在的陷阱!  

  

信息发送到服务器后,我们调用getInputStream(),并把返回值封装到一个DataInputStream 里,以便自己 

能读取结果。要注意的一件事情是结果以文本行的形式显示在一个TextArea (文本区域)中。为什么不简单 

地使用 getAppletContext()。showDocument(u)呢?事实上,这正是那些陷阱中的一个。上述代码可以很好地 

工作,但假如试图换用 showDocument(),几乎一切都会停止运行。也就是说,showDocument()确实可以运 

行,但从POSTtest 得到的返回结果是“Zero CONTENT_LENGTH”(内容长度为零)。所以不知道为什么原 

因,showDocument()阻止了 POST 查询向CGI 程序的传递。我很难判断这到底是一个在以后版本里会修复的 

Bug,还是由于我的理解不够(我看过的书对此讲得都很模糊)。但无论在哪种情况下,只要能坚持在文本区 

域里观看自 CGI 程序返回的内容,上述程序片运行时就没有问题。  



15。7 用 JDBC 连接数据库  



据估算,将近一半的软件开发都要涉及客户(机)/服务器方面的操作。Java 为自己保证的一项出色能力就 

是构建与平台无关的客户机/服务器数据库应用。在Java 1。1 中,这一保证通过 Java 数据库连接(JDBC ) 

实现了。  

数据库最主要的一个问题就是各家公司之间的规格大战。确实存在一种“标准”数据库语言,即“结构查询 

语言”(SQL…92),但通常都必须确切知道自己要和哪家数据库公司打交道,否则极易出问题,尽管存在所 

谓的“标准”。JDBC 是面向“与平台无关”设计的,所以在编程的时候不必关心自己要使用的是什么数据库 

产品。然而,从JDBC 里仍有可能发出对某些数据库公司专用功能的调用,所以仍然不可任性妄为。  

和Java 中的许多API 一样,JDBC 也做到了尽量的简化。我们发出的方法调用对应于从数据库收集数据时想 

当然的做法:同数据库连接,创建一个语句并执行查询,然后处理结果集。  

为实现这一“与平台无关”的特点,JDBC 为我们提供了一个“驱动程序管理器”,它能动态维护数据库查询 

所需的所有驱动程序对象。所以假如要连接由三家公司开发的不同种类的数据库,就需要三个单独的驱动程 

序对象。驱动程序对象会在装载时由“驱动程序管理器”自动注册,并可用Class。forName()强行装载。  

为打开一个数据库,必须创建一个“数据库URL”,它要指定下述三方面的内容:  

(1) 用“jdbc”指出要使用JDBC 。  

(2)  “子协议”:驱动程序的名字或者一种数据库连接机制的名称。由于JDBC 的设计从ODBC 吸收了许多灵 

感,所以可以选用的第一种子协议就是“jdbc…odbc 桥”,它用“odbc”关键字即可指定。  

(3) 数据库标识符:随使用的数据库驱动程序的不同而变化,但一般都提供了一个比较符合逻辑的名称,由 

数据库管理软件映射(对应)到保存了数据表的一个物理目录。为使自己的数据库标识符具有任何含义,必 

须用自己的数据库管理软件为自己喜欢的名字注册(注册的具体过程又随运行平台的不同而变化)。  

所有这些信息都统一编译到一个字串里,即“数据库URL”。举个例子来说,若想通过ODBC 子协议同一个标 



                                                                             576 


…………………………………………………………Page 578……………………………………………………………

识为“people”的数据库连接,相应的数据库URL 可设为:  

String dbUrl = 〃jdbc:odbc:people〃  

如果通过一个网络连接,数据库URL 也需要包含对远程机器进行标识的信息。  

准备好同数据库连接后,可调用静态方法 DriverManager。getConnection(),将数据库的URL 以及进入那个 

数据库所需的用户名密码传递给它。得到的返回结果是一个Connection 对象,利用它即可查询和操纵数据 

库。  

下面这个例子将打开一个联络信息数据库,并根据命令行提供的参数查询一个人的姓(Last Name)。它只选 

择那些有 E…mail 地址的人的名字,然后列印出符合查询条件的所有人:  

  

//: Lookup。java  

// Looks up email addresses in a   

// local database using JDBC  

import java。sql。*;  

  

public class Lookup {  

  public static void main(String'' args) {  

    String dbUrl = 〃jdbc:odbc:people〃;  

    String user = 〃〃;  

    String password = 〃〃;  

    try {  

      // Load the driver (registers itself)  

      Class。forName(  

        〃sun。jdbc。odbc。JdbcOdbcDriver〃);  

      Connection c = DriverManager。getConnection(  

        dbUrl; user; password);  

      Statement s = c。createStatement();  

      // SQL code:  

      ResultSet r =   

        s。executeQuery(  

          〃SELECT FIRST; LAST; EMAIL 〃 +  

          〃FROM people。csv people 〃 +  

          〃WHERE 〃 +  

          〃(LAST='〃 + args'0' + 〃') 〃 +  

          〃 AND (EMAIL Is Not Null) 〃 +  

          〃ORDER BY FIRST〃);  

      while(r。next()) {  

        // Capitalization doesn't matter:  

        System。out。println(  

          r。getString(〃Last〃) + 〃; 〃   

          + r。getString(〃fIRST〃)  

          + 〃: 〃 + r。getString(〃EMAIL〃) );  

      }  

      s。close(); // Also closes ResultSet  

    } catch(Exception e) {  

      e。printStackTrace();  

    }  

  }  

} ///:~  

  

可以看到,数据库URL 的创建过程与我们前面讲述的完全一样。在该例中,数据库未设密码保护,所以用户 

名和密码都是空串。  

用DriverManager。getConnection()建好连接后,接下来可根据结果Connection 对象创建一个Statement 



                                                                                          577 


…………………………………………………………Page 579……………………………………………………………

 (语句)对象,这是用createStatement()方法实现的。根据结果 Statement,我们可调用 

executeQuery(),向其传递包含了SQL…92 标准SQL 语句的一个字串(不久就会看到如何自动创建这类语句, 

所以没必要在这里知道关于SQL 更多的东西)。  

executeQuery()方法会返回一个ResultSet (结果集)对象,它与继承器非常相似:next()方法将继承器移 

至语句中的下一条记录;如果已抵达结果集的末尾,则返回null 。我们肯定能从executeQuery()返回一个 

ResultSet 对象,即使查询结果是个空集(也就是说,不会产生一个违例)。注意在试图读取任何记录数据 

之前,都必须调用一次 next()。若结果集为空,那么对next()的这个首次调用就会返回false。对于结果集 

中的每条记录,都可将字段名作为字串使用(当然还有其他方法),从而选择不同的字段。另外要注意的是 

字段名的大小写是无关紧要的——SQL 数据库不在乎这个问题。为决定返回的类型,可调用getString(), 

getFloat()等等。到这个时候,我们已经用Java 的原始格式得到了自己的数据库数据,接下去可用 Java 代 

码做自己想做的任何事情了。  



15。7。1 让示例运行起来  



就JDBC 来说,代码本身是很容易理解的。最令人迷惑的部分是如何使它在自己特定的系统上运行起来。之所 

以会感到迷惑,是由于它要求我们掌握如何才能使JDBC 驱动程序正确装载,以及如何用我们的数据库管理软 

件来设置一个数据库。  

当然,具体的操作过程在不同的机器上也会有所区别。但这儿提供的在32 位Windows 环境下操作过程可有效 

帮助大家理解在其他平台上的操作。  

  

1。 步骤 1:寻找JDBC 驱动程序  

上述程序包含了下面这条语句:  

Class。forName(〃sun。jdbc。odbc。JdbcOdbcDriver〃);  

这似乎暗示着一个目录结构,但大家不要被它蒙骗了。在我手上这个 JDK 1。1 安装版本中,根本不存在叫作 

JdbcOdbcDriver。class 的一个文件。所以假如在看了这个例子后去寻找它,那么必然会徒劳而返。另一些人 

提供的例子使用的是一个假名字,如“myDriver。ClassName ”,但人们从字面上得不到任何帮助。事实上, 

上述用于装载jdbc…odbc 驱动程序(实际是与JDK 1。1 配套提供的唯一驱动)的语句在联机文档的多处地方 

均有出现(特别是在一个标记为“JDBC…ODBC Bridge Driver ”的页内)。若上面的装载语句不能工作,那么 

它的名字可能已随着Java 新版本的发布而改变了;此时应到联机文档里寻找新的表述方式。  

若装载语句出错,会在这个时候得到一个违例。为了检验驱动程序装载语句是不是能正常工作,请将该语句 

后面直到catch 从句之间的代码暂时设为注释。如果程序运行时未出现违例,表明驱动程序的装载是正确 

的。  

  

2。 步骤2:配置数据库  

返回目录 上一页 下一页 回到顶部 1 1
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部!
温馨提示: 温看小说的同时发表评论,说出自己的看法和其它小伙伴们分享也不错哦!发表书评还可以获得积分和经验奖励,认真写原创书评 被采纳为精评可以获得大量金币、积分和经验奖励哦!