Java常见问题集锦

开发 后端
这篇文章是来自Sun中国官方网站关于一些Java常见问题集锦。希望这里与大家分享一下。

  Java常见问题集锦:

  问: 如何设置Java 2(JDK1.2)的环境变量?

  答:

    Java 2安装后,需要设置PATH和JAVA_HOME环境变量.与JDK1.1不同的是:设置好JAVA_HOME环境变量后,JVM将自动搜索系统类库以及用户的当前路径.

    Java 2环境变量的设置如下例所示:

    Solaris平台: setenv JAVA_HOME Java2的安装路径

    setenv PATH $JAVA_HOME/bin:${PATH}

    Windows平台: set JAVA_HOME=Java2的安装路径

    set PATH=$JAVA_HOMEbin;%PATH%

  问: 哪些Java集成开发工具支持Java 2?

  答:

    目前流行的Java集成开发环境,如Inprise的JBuilder,Symantec的Visual Cafe, Sybase的PowerJ,都支持Java 2.

  问: 如果在Netscape或IE浏览器中运行Java applet时出现了错误,如何确定错误范围?

  答:

    当java applet在浏览器中运行时,使用的是浏览器本身的缺省JVM.而不同浏览器对JDK的支持程度也不尽相同. 因此,在Netscape或IE浏览器中运行Java applet出现了错误,建议使用JDK提供的工具appletviewer或Sun公司的Hotjava浏览器来测试该applet,以确定错误的产生是与浏览器相关.

    如果applet在appletviewer或Hotjava中运行一切正常,则错误的产生是由于浏览 器不完全兼容JDK而引起的. 此时,解决方法可以是使用Hotjava浏览器或者安装 Sun公司的Java Plugin.

    如果applet在Hotjava浏览器或appletviewer中运行即发生错误,则应当根据错误 提示检查applet程序.

  问: 当用JDBC向数据库中插入数据或从数据库中提取数据时,为何有时中文字符会显示为乱码?

  答:

     这个问题的实现通常与各个JDBC driver的实现有关. 目前大多数JDBC driver采用本地编码格式来传输中文字符,例如中文字符"0x4175"会被转成"0x41"和"0x75"进行传输. 因此我们需要对JDBC driver返回的字符以及要发给JDBC driver的字符进行转换.

    当用JDBC driver向数据库中插入数据时,需要先将Unicode转成native code; 当 JDBC driver从数据库中查询数据时,则需要将native code转换成Unicode. 下面给出了这两种转换的实现:

  1. String native2Unicode(String s) {  
  2.   if (s == null || s.length() == 0) { 
  3.   return null;   
  4.   }   
  5.   byte[] buffer = new byte[s.length()];   
  6.   for (int i = 0; i s.length(); i++) { if (s.charAt(i)>= 0x100) {   
  7.   c = s.charAt(i);   
  8.   byte []buf = (""+c).getBytes();   
  9.   buffer[j++] = (char)buf[0];   
  10.   buffer[j++] = (char)buf[1];   
  11.   }   
  12.   else {   
  13.   buffer[j++] = s.charAt(i);   
  14.   }   
  15.   }   
  16.   return new String(buffer, 0, j);   
  17.   }  

    除使用以上两个方法之外,有些JDBC driver如果对jdbc driver Manager设置了正确 的字符集属性,以上2个方法就不需要了。

  问:当用Servlet来处理http请求并产生返回的HTML页面时,如何使HTML页面中的中文字符能够正常显示?

  答:

    javax.servlet.http.HttpResponse类用于产生返回页面.通过HttpResponse定义的方法getOutputStream()可以获得ServletOutputStream的实例,这样用户就可以利用ServletOutputStream.write方法向输出流中写入返回页面的内容. 但是ServletOutputStream使用的是缺省的编码方式,如果要使返回页面中的中文字 符能够正常显示,最好显示地指定所用的字符编码方式. 通常需要构造一个 OutputStreamWriter , 例程如下:

  1. public void doGet (HttpServletRequest req, HttpServletResponse res)   
  2.   throws ServletException, IOException   
  3.   {   
  4.   res.setContentType("text/html");   
  5.   ServletOutputStream out = res.getOutputStream();   
  6.   OutputStreamWriter ow = new OutputStreamWriter(out,"GB2312");   
  7.   ow.write("这是测试");   
  8.   ow.flush();   
  9.   ow.close();   
  10.   }      

  问:如何设置Java WebServer的CLASSPATH,以包含用户的class文件?

  答:

    有两种方法可以设置Java WebServer的CLASSPATH环境变量,以使用户编写的Servlet能够调用用户的class文件.

    将用户的class文件放到 JavaWebServer_Dir/classes目录下,在Java WebServer 启动时,classes目录被自动加入到CLASSPATH环境变量中了.

    修改httpd.nojre文件,将用户class文件所在的路径名加到CLASSPATH环境变量中.

  问:为什么在Windows平台上用Naming.lookup来获取远程RMI对象时会很慢?

  答:

    机器的网络设置不正确很可能会引起该问题的发生.

    RMI使用了Java网络类,特别是java.net.InetAddress类,它将查询TCP/IP的主机名, 包括IP地址到主机名的映射和主机名到IP地址的映射.在Windows平台,这种查询功能 是由本地的Windows Socket库来实现的. 因此延时是发生在Windows库中,而非RMI中.

    如果你的机器设置成使用DNS,问题通常是DNS服务器查不到主机名,你所发现的延时 是DNS查询的延时. 请尝试将RMI通信中涉及到的所有主机名/IP地址加到本地文件winntsystem32driversetchosts或windowshosts中. 格式如下:

    IP地址 主机名

    如此设置应当可以明显地减少查询所花的时间.

  问: 编写Java application时,如何设置proxy的信息,以便访问外部网站?

  答:

    若在java application中访问外部网站,首先应设置proxy信息,样例代码如下:

  1. import java.util.properties;   
  2.   .....   
  3.   Properties sys = System.getProperties();   
  4.   sys.put("proxySet","true");   
  5.   sys.put("proxyHost","myHTTP.proxyserver.com");   
  6.   sys.put("proxyPort","80");   
  7.   System.setProperties(sys);   
  8.   u = new URL(website);   
  9.   connect = (HttpURLConnection)u.openConnection();   
  10.   .....      

  问: Swing组件JList的列表数据修改了,如何通知JList改变显示?

  答:

    JList组件有一个单独的显示模式ListModel来表示JList的显示数据.

    JList创建以后,JList数据元素的值及数据元素的数量可以动态地改变.

    JList在它的数据模式ListModel中观察数据的改变.因此,一个ListModel 的正确实现应当在每次数据发生改变时,通知事件的监听者.

    当使用构造函数JList(Object[])创建一个JList的实例时,系统将自动 创建一个DefaultListModel的实例来存储JList的显示数据, 可以调用 DefaultListModel中定义的简便方法来动态地修改JList的数据,如 removeElementAt(index),addElement(Object)等. DefaultListModel 在修改数据的同时,将通知JList关于数据的改变.

问:在Java applet中如何实现一个模式对话框?

  答:

    在Java applet中实现模式对话框的关键就是在创建一个对话框的时候 要为该对话框指定一个正确的父窗口.因为Applet是Panel类的子类,不 可以作为对话框的父窗口,所以首先要获得applet所在的窗口,作为模式 对话框的父窗口. 样例代码如下:

  1. .....   
  2.   Dialog d = new Dialog( getParentWindow(comp),title);   
  3.   // comp为applet上的任意一个组件   
  4.   ....   
  5.   public void getParentWindow(Component compOnApplet,String title){   
  6.   Container c = compOnApplet.getParent();   
  7.   while (c != null) {   
  8.   if (c instanceof Frame)   
  9.   return (Frame) c;   
  10.   c = c.getParent();  
  11.   }   
  12.   return null;   
  13.   }      

问: 在Java applet中如何显示另外一个HTML页面?

  答:

    通过java.applet.Applet.getAppletContext()方法可以获得与该applet相关的AppletContext, AppletContext.showDocument(URL)方法就可以使applet所在的浏览器显示另外一个网页.

  问: 用JDK实现的签名applet,可否在Netscape或IE中运行?

  答:

    用JDK实现的签名applet,不可以在Netscape或IE中运行,但是可以在Hotjava浏览器中运行.

    不同的浏览器提供了不同的签名applet机制,如Netscape提供了zigbert工具和 Capability API, 而IE则需要使用CAB文件. 但是,无论是Netscape工具产生的 签名applet,还是用IE产生的签名applet,都不可以在其它的浏览器中运行.

    如果要使JDK产生的签名applet能够在Netscape或IE中运行,解决方法是在 Netscape或IE中安装Java Plugin,则用JDK实现的签名applet就可以在这两种 浏览器中运行.

问: 用JNI技术可以从Java应用中调用C程序库,但是如何使该C程序库可以调用另外的C程序库?

  答:

    如果一个被Java调用的C程序库C1仍需要调用另外一个C程序库C2,那么在编译C1的时候应当联接程序库C2,步骤如下(Solaris平台):

    编写调用C库的Java文件,并编译.

    javac java文件名

    产生C程序头文件

    javah -jni java文件名(不带后缀.java)

    编写被Java调用的C程序C1.c,以及被C1调用的C2.c,并编译.

    cc -G -Iinclude路径名 C2.c -o libC2.so

    cc -G -Iinclude路径名 -lC2 C1.c -o libC1.so

    设置环境变量

    setenv LD_LIBRARY_PATH libC1.so,libC2.so所在路径:${LD_LIBRARY_PATH}

    运行java应用

#p#

  问: 在Java语言中,如何列出PC机文件系统中的所有驱动器名?

  答:

    在Java 2版本中,java.io包中的File类新增加了方法listRoots()可以实现这一功能.

问: 为什么Runtime.exec("ls")没有任何输出?

  答:

    调用Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息. 由于调用Runtime.exec方法所创建的子进程没有自己的终端或控制台,因此该子进程的标准IO(如stdin,stdou,stderr)都通过Process.getOutputStream(),Process.getInputStream(), Process.getErrorStream()方法重定向给它的父进程了.用户需要用这些stream来向 子进程输入数据或获取子进程的输出. 所以正确执行Runtime.exec("ls")的例程如下:

  1. try  
  2.   { 
  3.   process = Runtime.getRuntime().exec (command);   
  4.   InputStreamReader ir=newInputStreamReader(process.getInputStream());   
  5.   LineNumberReader input = new LineNumberReader (ir);   
  6.   String line;   
  7.   while ((line = input.readLine ()) != null)   
  8.   System.out.println(line);   
  9.   }   
  10.   catch (java.io.IOException e){   
  11.   System.err.println ("IOException " + e.getMessage());   
  12.   }      

问: 如何产生签名applet,以使applet能够访问本地资源?

  答:

    在jdk1.1中,可以使用javakey命令来产生公钥,私钥,证书和签名的jar文件,详细资料 请参考: http://java.sun.com/security/usingJavakey.html而java 2对签名机制做了比较大的改进,允许用户更灵活地设置安全权限.Java 2提供了三个工具:keytool,policytool和jarsigner来实现签名applet.例如,Joe编写了一个签名applet:SignedApplet.java,那么产生一个简单的签名applet的过程如下:

  //产生密钥,密钥别名为joe,口令为sign12,存放在密钥库joestore中

  keytool -genkey -alias joe -keypass sign12 -keystore joestore

  //将SignedApplet.class及相关文件打包成jar文件

  jar cvf SignedAppletDemo.jar

  //利用keytool生成的自签名的证书产生签名applet(jar文件)

  jarsigner -keystore joestore -signedjar joe.jar SignedAppletDemo.jar joe

  //将自签名证书从keystore中输出到文件

  keytool -export -keystore joestore -alias joe -file joe.cer

  而对于签名applet的接受方Susan,需要通过如下步骤来安全地执行

  Joe编写的签名applet:

  //得到Joe的证书并将之读入到密钥库中susanstore中

  keytool -import -alias joe -file joe.cer -keystore susanstore

  //运行policytool产生满足Susan要求的policy文件

  policytool

  //用appletviewer运行之,或在浏览器中安装java plugin来运行之.

  关于签名applet在Java Plugin中的部署请参考以下网页: http://java.sun.com/security/signExample12/

  注:以上的例子为简单起见,使用了keytool产生的自签名证书.其实,用户也可以使用keytool -certreq向商业CA中心申请电子证书.

  问:若通过ObjectOutputStream向一个文件中多次以追加方式写入object,为什么用ObjectInputStream读取这些object时会产生StreamCorruptedException?

  答:

    使用缺省的serializetion的实现时,一个ObjectOutputStream的构造和一个ObjectInputStream的构造必须一一对应.ObjectOutputStream的构造函数会向输出流中写入一个标识头,而ObjectInputStream会首先读入这个标识头.因此,多次以追加方式向一个文件中写入object时,该文件将会包含多个标识头.所以用ObjectInputStream来deserialize这个ObjectOutputStream时,将产生StreamCorruptedException.一种解决方法是可以构造一个ObjectOutputStream的子类,并覆盖writeStreamHeader()方法.被覆盖后的writeStreamHeader()方法应判断是否为首次向文件中写入object,羰?则调用super.writeStreamHeader();若否,即以追加方式写入object时,则应调用ObjectOutputStream.reset()方法.

  问: 对象的序列化(serialization)类是面向流的,应如何将对象写入到随机存取文件中?

  答:

    目前,没有直接的方法可以将对象写入到随机存取文件中,但是可以使用ByteArray输入/输出流作为中介,来向随机存取文件中写入或从随机存取文件中读出字节,并且可以利用字节流来创建对象输入/输出流,以用于读写对象.需要注意的是在字节流中要包含一个完整的对象,否则读写对象时将发生错误. 例如,java.io.ByteArrayOutputStream可用于获取ObjectOutputStream的字节流,从中可得到byte数组并可将之写入到随机存取文件中.相反,我们可以从随机存取文件中读出字节数组,利用它可构造ByteArrayInputStream,进而构造出ObjectInputStream,以读取对象.

  问: 运行RMI应用时,可不可以不手工启动名字服务rmiregistry,而是从程序中启动之?

  答:

    可以. java.rmi包中提供了类java.rmi.registry.LocateRegistry,用于获取名字服务或创建名字服务.调用LocateRegistry.createRegistry(int port)方法可以在某一特定端口创建名字服务,从而用户无需再手工启动rmiregistry.此外,LocateRegistry.getRegistry(String host,int port)方法可用于获取名字服务.

  问: 使用类PrintJob进行打印操作时,应如何设置打印机名等打印属性?

  答:

    使用如下方法可以获得PrintJob的实例用于控制打印操作:

    Toolkit.getPrintJob(Frame f, String jobtitle, Properties prop)

    那么对于打印属性的设置可以通过对prop的属性设置来实现,打印属性包括:

    awt.print.destination: 可以是"printer"或"file"

    awt.print.printer: 打印机名

    awt.print.fileName: 打印文件名

    awt.print.numCopies: 打印份数

    awt.print.options: 打印命令的打印选项

    awt.print.orientation: 打印方向,可以是"portrait"或"landscape"

    awt.print.paperSize: 纸张大小,可以是"letter","legal","executive"或"a4"

问:在JDK1.1中Thread类定义了suspend()和resume()方法,但是在JDK1.2中已经过时,应使用什么方法来替代之?

  答:

    Thread.suspend本身易于产生死锁.如果一个目标线程对某一关键系统资源进行了加锁操作,然后该线程被suspend,那么除非该线程被resume,否则其它线程都将无法访问该系统资源.如果另外一个线程将调用resume,使该线程继续运行,而在此之前,它也需要访问这一系统资源,则将产生死锁.

    因此,在Java 2中,比较流行的方式是定义线程的状态变量,并使目标线程轮询该状态变量,当状态为悬挂状态时,可以使用wait()方法使之处于等待状态.一旦需要该线程继续运行,其它线程会调用notify()方法来通知它.

  问: 使用JDBC编程,应如何控制结果集ResultSet的指针,使之能够上下移动,以及移动到结果集的第一行和最后一行?

  答:

    在JDK1.1中,ResultSet类中只定义了next()方法支持数据指针的下移.但在Java 2中,ResultSet类增加了如下方法支持数据指针的移动,包括:

    ResultSet.first():将数据指针移到结果集的第一行

    ResultSet.last(): 将数据指针移到结果集的最后一行

    ResultSet.previous(): 将数据指针上移一行

    以上的方法定义在JDBC2.0的规范中,所有支持JDBC 2.0的JDBC驱动程序都可以支持上述方法.目前Intersolv和OpenLink等JDBC驱动程序厂商均有产品支持JDBC 2.0 .

  问: 哪几种Web Server支持Servlet?如何使IIS支持Servlet?

  答:

    目前,支持Servlet的服务器端产品主要有: Sun公司的Java WebServer,Lotus DominoGo WebServer,BEA weblogic Tengah Server,Jigsaw,NetForge,AcmeServer和Mot Bays Jetty等.

    此外,一些第三方厂商也开发了Servlet engine,以使其它WebServer(如Netscape Web Server,IIS等)能够运行Servlet,如LiveSoftware的Jrun(http://www.livesoftware.com/ products/jrun/)等.

  问: 如何在Java应用中将图像存储到图像文件中?

  答:

    Java Advanced Imaging API(包含在Java Media API中)允许在Java应用中执行复杂的,高性能的图像处理.JAI API提供了存储图像的能力.目前,JAI API支持以下几种图像文件格式:BMP,JEPG,PNG,PNM,TIFF.下面给出了将图像存储到BMP文件的一段代码:

  1. OutputStream os = new FileOutputStream(fileToWriteTo);   
  2.   BMPEncodeParam param = new BMPEncodeParam();   
  3.   ImageEncoder enc = ImageCodec.createImageEncoder("BMP", os, param);   
  4.   enc.encode(img);   
  5.   os.close();      

  有关存储图像文件的编程指南请参考以下网页:

  http://java.sun.com/products/java-media/jai/forDevelopers/jai-guide/

  问: 如何用Java语言向串口读写数据?

  答:

    Sun公司的Java Communication API2.0可用于读写串口,它支持RS232串口和IEEE 1284 并口,提供了一种与平台无关的串/并口通信机制.

【编辑推荐】

  1. 浅谈Java学习方法
  2. 高手支招 Java经验分享(一)
  3. Java基础知识的三十个经典问答
  4. 新手入门:学习Java的一点经验心得
  5. 必读!Java开发人员的十大戒律
责任编辑:韩亚珊 来源: 电脑编程技巧与维护杂志社
相关推荐

2010-07-21 09:10:02

Perl常见问题

2011-05-07 17:16:05

复印机常见问题

2012-02-06 10:37:07

Java

2013-11-14 15:47:29

SDN问题答疑

2011-05-06 15:39:55

硒鼓

2010-05-13 13:27:23

2009-11-02 17:25:04

ADSL常见问题

2010-03-25 09:08:43

CentOS配置

2011-02-22 14:00:16

vsftpd

2009-12-31 09:58:51

Ubuntu常见问题

2010-04-14 14:37:56

Oracle卸载

2010-07-21 09:16:47

Perl常见问题

2009-09-22 09:22:03

.NET常见问题

2013-08-13 13:16:02

Android adb问题

2011-03-29 13:23:54

CACTI

2011-04-08 13:58:52

JavaJSP

2009-11-09 10:42:53

ibmdwRational

2011-04-07 09:05:19

Ubuntu 11.0

2010-04-23 09:58:30

Oracle管理

2010-10-12 16:50:31

综合布线
点赞
收藏

51CTO技术栈公众号