数学建模社区-数学中国

标题: [转帖]Java常见问题集锦(来自Sun中国官方站) [打印本页]

作者: god    时间: 2005-3-31 01:31
标题: [转帖]Java常见问题集锦(来自Sun中国官方站)
问: 如何设置Java 2(JDK1.2)的环境变量?  + R3 F! y  W$ |8 f( G( m6 f/ _) K
) t3 f" z& j& z- w/ X
答: Java 2安装后,需要设置PATH和JAVA_HOME环境变量.与JDK1.1不同的是:设置好JAVA_HOME环境变量后,JVM将自动搜索系统类库以及用户的当前路径.  / X: e" `3 u- s' _
Java 2环境变量的设置如下例所示:  
6 v5 R4 E) w2 @/ n$ dSolaris平台: setenv JAVA_HOME Java2的安装路径  
8 i8 ^" B& h9 ?' e1 ~/ Rsetenv PATH $JAVA_HOME/bin{PATH}  3 j$ h  j. G3 a0 O
Windows平台: set JAVA_HOME=Java2的安装路径  7 Z) p9 t3 m- o: m
set PATH=$JAVA_HOMEbin;%PATH%  
5 h. i8 p5 G/ u' Y0 \
& Y& h" O* j# |* X! Y* O$ p3 I& q# }问: 哪些Java集成开发工具支持Java 2?  8 n* \$ }5 Y& f* v) f
: l# s4 ]5 v: s
答: 目前流行的Java集成开发环境,如Inprise的JBuilder,Symantec的Visual Cafe, Sybase的PowerJ,都支持Java 2.  
4 C3 I9 _( Y5 I1 J4 b5 C
- r3 X0 r  x: [7 C问: 如果在Netscape或IE浏览器中运行Java applet时出现了错误,如何确定错误范围?  
1 q9 x# X- ~( |. g
9 P) Q9 B/ {$ M& u! Y答: 当java applet在浏览器中运行时,使用的是浏览器本身的缺省JVM.而不同浏览器对JDK的支持程度也不尽相同. 因此,在Netscape或IE浏览器中运行Java applet出现了错误,建议使用JDK提供的工具appletviewer或Sun公司的Hotjava浏览器来测试该applet,以确定错误的产生是与浏览器相关.  
5 \4 F7 c  ~7 T; q6 A3 e    如果applet在appletviewer或Hotjava中运行一切正常,则错误的产生是由于浏览 器不完全兼容JDK而引起的. 此时,解决方法可以是使用Hotjava浏览器或者安装 Sun公司的Java Plugin.  
8 ?2 h5 Y$ I% Q' w    如果applet在Hotjava浏览器或appletviewer中运行即发生错误,则应当根据错误 提示检查applet程序.  6 e1 C8 d! M6 x' i1 P$ v, R

- n0 H! e; A9 P" D2 g- W" H- V1 Q& L问: 当用JDBC向数据库中插入数据或从数据库中提取数据时,为何有时中文字符会显示为乱码?  
+ o- F1 O! y; h1 q2 k/ x; X# Y' a& u- h
答: 这个问题的实现通常与各个JDBC driver的实现有关. 目前大多数JDBC driver采用本地编码格式来传输中文字符,例如中文字符"0x4175"会被转成"0x41"和"0x75"进行传输. 因此我们需要对JDBC driver返回的字符以及要发给JDBC driver的字符进行转换.    _3 d4 }0 ?6 P) g0 W4 i0 [- s
3 @; p! n' k# r/ O. I
当用JDBC driver向数据库中插入数据时,需要先将Unicode转成native code; 当 JDBC driver从数据库中查询数据时,则需要将native code转换成Unicode. 下面给出了这两种转换的实现:  9 R7 ^8 h, K4 n0 R1 P  \
9 q! J/ d0 [/ c. I! `
& E, x7 f; Q+ a9 S( u, |. M* Q
String native2Unicode(String s) {  
# z  }' ?  R5 W! L( d+ h1 _" Y* ~' M2 p, Z7 P( F$ a, U# {0 O/ s1 ~) R
if (s == null || s.length() == 0) {  
, R9 S3 O& C3 m5 V% P6 u6 }& ~/ y5 Y4 B+ b' R% Y- y8 q
return null;  & f/ W7 R# o: C3 v7 o6 c3 b% T
+ v% R/ j  R% f1 s
}  % \9 e( ^" Y% `: E

8 G& G) z1 O8 m; a! o. ?byte[] buffer = new byte[s.length()];  
/ s3 M% s5 k' n3 s# d
: i2 J( ]: s; E* k* }for (int i = 0; i s.length(); i++) { if (s.charAt(i)>= 0x100) {  % O( B; H, K+ ]% U
; V; W/ X. J: @- w& M
c = s.charAt(i);  
' m- n. Q- `% |! t" G4 W- B& F; a0 x
8 L( F" T; {. {5 I) Abyte []buf = (""+c).getBytes();  
. n$ h' H" X/ l& o$ X' i* p( E. I. t+ T' G% k7 r& g" d7 E3 @; S
buffer[j++] = (char)buf[0];  ' K; W8 l! M( X2 h: x
% j: b% I7 E! Q( B+ i6 e# G3 I7 b
buffer[j++] = (char)buf[1];  ( G# c( O" k" d) D+ D% Z
6 T& m0 Z9 P$ V% E
}  4 v2 S6 L. e5 V( u: k; t

8 ^1 Q' N/ D- e  _3 b7 ~else {  
9 [) w% q4 P. S1 K) V- O  R7 G
0 s: c$ d- r) w/ }( gbuffer[j++] = s.charAt(i);  
4 \- J: q( P# ^/ s& D$ y- E2 g, U2 Q' J& H+ k( P, s8 @
}  / j& z, ?) q: N+ x- D, n& _0 @- j
: o% [3 I( [) u, N" |
}  
( n, C; z( E5 @  B8 J2 e
% N! t* A% H8 N1 I) P% Nreturn new String(buffer, 0, j);  
7 @5 d: @7 f) _9 S+ g2 P# w; s  R* W, k$ D8 w. @* F
}  
7 i8 ]0 q' D% f+ _* o" i0 p. u) a) K, x  ?; _1 u: R& J
除使用以上两个方法之外,有些JDBC driver如果对jdbc driver Manager设置了正确 的字符集属性,以上2个方法就不需要了.  7 b/ }- h' }7 _

( X6 L; S" _# h* \
$ d0 N4 f# d: s4 W* K9 F$ S/ t' x4 X问:  ; y( o! M3 M/ M  \# H2 k
当用Servlet来处理http请求并产生返回的HTML页面时,如何使HTML页面中的中文字符能够正常显示?  7 e# V# c: B( s" L& I

0 x5 ^8 @- O$ e' K: a. K答:  
  e; _- \8 W1 njavax.servlet.http.HttpResponse类用于产生返回页面.通过HttpResponse定义的方法getOutputStream()可以获得ServletOutputStream的实例,这样用户就可以利用ServletOutputStream.write方法向输出流中写入返回页面的内容. 但是ServletOutputStream使用的是缺省的编码方式,如果要使返回页面中的中文字 符能够正常显示,最好显示地指定所用的字符编码方式. 通常需要构造一个 OutputStreamWriter , 例程如下:  
" \9 n+ `. Z6 M/ f, G- d- A
! x9 u. T2 l1 y6 M: }
* S, e- ^/ ]+ i2 ipublic void doGet (HttpServletRequest req, HttpServletResponse res)  
  x5 l4 S+ p, i
$ D+ z3 Y" z4 D" K# ^4 D8 Gthrows ServletException, IOException  " h( I& Y) i1 S3 A& M5 ]

8 f6 A% S' S+ H0 B9 e7 T- c{  . x: L, d5 u4 b3 m

4 o( O  y+ y" Q# U; X0 Vres.setContentType("text/html");  
  B% n/ W# M& S8 Z! V
8 ~6 M* J  U" p) ?! |/ JServletOutputStream out = res.getOutputStream();  + D: E: a( x6 b$ C: g7 `3 T
4 A/ P8 `# f4 K
OutputStreamWriter ow = new OutputStreamWriter(out,"GB2312");  4 n  [% |8 U/ S* Q, [, P1 C5 d
/ v2 M$ O* r1 q( h& a% b: ?/ c- l
ow.write("这是测试");  $ n' p  @( ^: t0 n

) J7 L: L, n; s6 Y0 wow.flush();  2 O' K  |% c' L
# |3 N; i, N2 p6 i' l# K* V; r
ow.close();  
& V/ `/ \3 h! t8 y! I% J0 B! d# y9 b' W# p0 ^; \* _, U8 p# W
}  - f, Y& Y$ c" a2 i2 U/ b) {
: t+ g# U2 r0 w$ u9 k* ]% ]7 G
: j, J6 x5 i6 P* D
问:  
$ ]- @, `/ T8 d& R, A! F如何设置Java WebServer的CLASSPATH,以包含用户的class文件?  + I! A1 d% K- s7 g! b2 ^2 k

0 K' f5 v+ {0 I/ [2 K答:  
3 z  [3 |  i' l  y有两种方法可以设置Java WebServer的CLASSPATH环境变量,以使用户编写的Servlet能够调用用户的class文件.  9 @' ~$ [& s* l: q5 g
将用户的class文件放到 JavaWebServer_Dir/classes目录下,在Java WebServer 启动时,classes目录被自动加入到CLASSPATH环境变量中了.  
7 M6 Z( z! Z% o# _' _) Q% H9 `修改httpd.nojre文件,将用户class文件所在的路径名加到CLASSPATH环境变量中.  * w5 U3 N4 N& j* ^5 w0 i  ]5 f
( L8 L4 |( O+ D5 K- Y1 M( L# @
; e) t/ X4 I; J" }' v
问:  
2 A9 c0 G* e# u为什么在Windows平台上用Naming.lookup来获取远程RMI对象时会很慢?  
- \% g2 Q0 {. e1 S# X3 G, J
5 z9 P6 \2 T, V( l答:  4 s, @7 A: f  N: M% |. h
机器的网络设置不正确很可能会引起该问题的发生.  ' y) q7 R8 H$ q! p
RMI使用了Java网络类,特别是java.net.InetAddress类,它将查询TCP/IP的主机名, 包括IP地址到主机名的映射和主机名到IP地址的映射.在Windows平台,这种查询功能 是由本地的Windows Socket库来实现的. 因此延时是发生在Windows库中,而非RMI中.  
* B9 n6 x% J) D4 h
( }8 B! B9 }( o; |8 n如果你的机器设置成使用DNS,问题通常是DNS服务器查不到主机名,你所发现的延时 是DNS查询的延时. 请尝试将RMI通信中涉及到的所有主机名/IP地址加到本地文件 winntsystem32driversetchosts或windowshosts中. 格式如下:  8 t# C1 \" M- z4 h- E

, l2 w# J4 d" Q" JIP地址 主机名  
" B0 R, A! t0 e; V+ `+ L* j9 i5 B/ ^- f4 `7 a) d. Q2 {7 D
如此设置应当可以明显地减少查询所花的时间.  7 h6 z% r7 q9 L# K3 H; _
4 O6 @1 j1 t: @" V+ @0 H
问: 编写Java application时,如何设置proxy的信息,以便访问外部网站?  
6 W" ]5 j" z  ]9 ]& H9 W. }' P, u5 V
" \% R' B$ M% F/ |# h2 y8 o答:  
" X' D8 b' t5 n" }/ ~, v" s( S3 E若在java application中访问外部网站,首先应设置proxy信息,样例代码如下:  7 y; e) H/ h4 \# B0 c
, v/ u9 e6 ]. v5 l9 u6 A
! \, R2 ?9 K" S; ]& v) _# n
import java.util.properties;  
% L  B. i3 Y. {$ t% o. B- e6 y' ?6 k2 S5 t
.....  9 v0 ~3 l8 q1 F; C  y

5 d/ n7 J3 ~5 r1 KProperties sys = System.getProperties();  
0 o$ b& D7 s' Q! i( \) c" i" Q! n: Y  ^5 b" ^% o7 U- H
sys.put("proxySet","true");  
6 x( l4 F0 V! B9 r3 a
% S, \- X1 O( L& P% k$ ]3 z. ysys.put("proxyHost","myHTTP.proxyserver.com");  
) f2 [- N, H+ z* E1 q& q" n+ n% M! {8 e  |
sys.put("proxyPort","80");  6 D: Z' ^* i' T
5 p- l( _5 [- D! E  X
System.setProperties(sys);  
3 X. O/ r/ k4 B' @$ H5 D# w5 f# F4 ]5 \

. Q" O) V! T: e. W: X& ?$ M% S7 C
; w) b1 _0 p0 au = new URL(website);  
( \+ X0 n. f/ G7 B, K
2 @2 V; I# [9 G  B* V  K5 v) Xconnect = (HttpURLConnection)u.openConnection();  
" q7 e0 j0 O7 H- w& u; A: k  p& b# {' d0 a# u  `2 Y; G& J
.....  # l5 y( \! e! E" c) ]6 W
' {& g( l+ ^; D: `
问: Swing组件JList的列表数据修改了,如何通知JList改变显示?  
) V$ s# j! ~" E5 U
8 N5 f* c: _9 n1 L: [5 X答:  
( b" j& ?# H: p! x. _' W: eJList组件有一个单独的显示模式ListModel来表示JList的显示数据.  
6 X0 `: [. J& }! X5 c9 }- h6 dJList创建以后,JList数据元素的值及数据元素的数量可以动态地改变.  
# S6 D8 p( w0 y- R" uJList在它的数据模式ListModel中观察数据的改变.因此,一个ListModel 的正确实现应当在每次数据发生改变时,通知事件的监听者.  . I6 M3 O$ o8 w* s! h! _
当使用构造函数JList(Object[])创建一个JList的实例时,系统将自动 创建一个DefaultListModel的实例来存储JList的显示数据, 可以调用 DefaultListModel中定义的简便方法来动态地修改JList的数据,如 removeElementAt(index),addElement(Object)等. DefaultListModel 在修改数据的同时,将通知JList关于数据的改变.  * n" f. |& W6 l* F' Y' Z- w' T, F
2 p$ G$ a% o% t8 o# K5 r
问:  
; J- C; Q2 U1 [7 f$ E在Java applet中如何实现一个模式对话框?  
2 p- t8 g9 l6 D2 L) a2 @5 Y6 @8 [
/ z2 }) l' B$ H: S2 Z答:  
% k$ G5 B1 h1 y在Java applet中实现模式对话框的关键就是在创建一个对话框的时候 要为该对话框指定一个正确的父窗口.因为Applet是Panel类的子类,不 可以作为对话框的父窗口,所以首先要获得applet所在的窗口,作为模式 对话框的父窗口. 样例代码如下:  
& V+ ]8 e" B" e# D# D( z* ]& g" t
& X* b. V6 f. I2 X3 N, u1 v4 t$ }% ^  }1 P/ r  P  z5 e
.....  
3 ]$ x  ~* ], u$ |; `( Z7 v# E  ]4 _+ x+ Z
Dialog d = new Dialog( getParentWindow(comp),title);  ' e' o/ v' b  O
  K, {- G* y. z
// comp为applet上的任意一个组件  ) ?2 r, d  T  C$ \6 `, ~8 O
7 ^5 p. d/ \$ A8 L
....  ( L+ b3 y4 r! i; K$ H2 g

8 l5 ^; p) F6 T1 h$ ?& w( z. t
2 s6 H) f8 q3 Y1 {" v& t0 G# y* P- E+ e
public void getParentWindow(Component compOnApplet,String title){  
3 |! p6 l/ i6 [* {4 ]: Y, q
& {4 x* k+ ~$ F# CContainer c = compOnApplet.getParent();  ! f' B) P2 k6 \

$ U( r" d7 z( o; _  wwhile (c != null) {  
2 A  W1 b# O+ e+ o  Z) _
" Y* `  k: h& l6 Iif (c instanceof Frame)  
& ?2 r. |# l) h3 a
0 o- c, g/ N$ y- a( N0 p' sreturn (Frame) c;  
1 s$ J3 S! z5 M; o* V
* [! i9 p/ c/ Lc = c.getParent();  & k: V- r+ t2 t
# {/ N) i' `+ x) }
}  * n* P+ e/ |0 e2 v
$ N" E+ [1 r* c8 P
return null;  
) r3 W+ a5 y3 `$ J5 C: m# N! f1 o
) T$ ]; Y: l7 K" o}  6 i* g5 p# [' q3 {+ ~  j

5 q9 }/ q4 E/ C' ]! I5 x$ s问: 在Java applet中如何显示另外一个HTML页面?  ' A3 j5 k; n9 L$ [2 b
4 A7 O& \/ Z9 w, [$ @8 A, C5 z
答:  
; O! @) z$ H% R通过java.applet.Applet.getAppletContext()方法可以获得与该applet相关的AppletContext, AppletContext.showDocument(URL)方法就可以使applet所在的浏览器显示另外一个网页.  
) U4 y. ^/ G& a' |# u+ g0 o  ]
2 d2 v, F3 }8 X4 U问:  " z% _2 j' x8 t# C' }' i9 V
用JDK实现的签名applet,可否在Netscape或IE中运行?  
& f, C8 x& Q8 H' w2 i* \/ y& L$ L) Y" U* [
答:  ) w9 x1 B, U$ f2 j( U/ m( ?
用JDK实现的签名applet,不可以在Netscape或IE中运行,但是可以在Hotjava浏览器中运行.  
7 ^/ l9 K% G" B1 Y9 T
  |! Q, Y& j, d! T; \4 I不同的浏览器提供了不同的签名applet机制,如Netscape提供了zigbert工具和 Capability API, 而IE则需要使用CAB文件. 但是,无论是Netscape工具产生的 签名applet,还是用IE产生的签名applet,都不可以在其它的浏览器中运行.  8 H. h% c% s, X/ b' ^/ i
  u8 M* S8 _$ d
如果要使JDK产生的签名applet能够在Netscape或IE中运行,解决方法是在 Netscape或IE中安装Java Plugin,则用JDK实现的签名applet就可以在这两种 浏览器中运行.  
2 A& R# u& C6 m7 I0 m8 i; j
, ?) @* F( p! ~# K. O$ v0 |. L问:  1 F& W" I# h  ?( S, ?" \0 L
用JNI技术可以从Java应用中调用C程序库,但是如何使该C程序库可以调用另外的C程序库?  
2 I" [/ p( a9 l+ `$ ^( n, d* ~" B- ^8 H$ X
答:    k8 S9 `. N- ]7 D' g$ K
如果一个被Java调用的C程序库C1仍需要调用另外一个C程序库C2,那么在编译C1的时候应当联接程序库C2,步骤如下(Solaris平台):  & B( U3 m* }2 T( ?7 R9 e% A  K$ l
编写调用C库的Java文件,并编译.  & p% P% {4 m2 w0 }  k
javac java文件名  
5 g3 b: Z$ `. m$ f  Z5 N! ^3 q
/ r2 I% L3 v. r, Y. s5 T/ U
; z! O* D/ l2 s; R产生C程序头文件  
0 a3 y7 w1 ]8 a' c! p7 I& v4 J) C3 l3 vjavah -jni java文件名(不带后缀.java)  # }: X4 U4 D! e! m) h) ]6 r( I

1 l: L* ~4 y% M9 i# s7 v* ~( T
  k5 Q( F1 a. M7 M/ v编写被Java调用的C程序C1.c,以及被C1调用的C2.c,并编译.  . J$ S7 Y2 M1 ^* R, i& s
cc -G -Iinclude路径名 C2.c -o libC2.so  
1 t( X- p* {& X0 ~' Tcc -G -Iinclude路径名 -lC2 C1.c -o libC1.so  ( g/ U& \% X/ b9 N, r
  a; d: a9 N/ C8 O7 n

; s  D; D& ~% k5 G设置环境变量  ) q/ J8 q3 X6 H( X; A! }
setenv LD_LIBRARY_PATH libC1.so,libC2.so所在路径  
. N; Q0 O5 ?5 G# n$ W- B{LD_LIBRARY_PATH}  
, a( G( {% O9 Q, ^5 w# ^
; P/ e. }% Z5 r% |# l. u; z" b; d$ R) `
运行java应用  
! R5 \$ l3 A( `$ |; O' b- c0 g; f' H6 b
) v) G3 A. Q- ]
问:  % O" P2 ?: m; A+ b
在Java语言中,如何列出PC机文件系统中的所有驱动器名?  : s6 J8 {, g& C( b  n5 d

3 S! m% l7 n$ @! _# |& ]( f" u答:  . B; ^; u0 m# A, W8 r
在Java 2版本中,java.io包中的File类新增加了方法listRoots()可以实现这一功能.  ) p9 t# ^$ @* C

! ]1 I; e  f  Y* A8 g5 p' t3 \问:  
2 P& y! W" M2 T( }' ?7 U为什么Runtime.exec("ls")没有任何输出?  
4 F( W4 O' j) u$ X) q
( {6 Y. ]7 D. H( s# l5 G答:  ' Y2 U, l3 z5 H7 S( q7 L) @5 ]
调用Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息. 由于调用Runtime.exec方法所创建的子进程没有自己的终端或控制台,因此该子进程的标准IO(如stdin,stdou,stderr)都通过Process.getOutputStream(),Process.getInputStream(), Process.getErrorStream()方法重定向给它的父进程了.用户需要用这些stream来向 子进程输入数据或获取子进程的输出. 所以正确执行Runtime.exec("ls")的例程如下:  
9 Q( }4 E, s' h) ^/ t( y2 {- Q0 W' g" A
4 v' r2 k0 @$ z2 I( B* U, g$ i3 ^  [/ V
try  
, M! j5 D8 W1 p
4 b' j$ Y' V% }$ a' d{  
( z3 \; Z& L+ |  E: N& P1 b/ D2 f# x2 \) z$ _2 C' ~- V% x
process = Runtime.getRuntime().exec (command);  
5 q$ ?, m2 F7 z$ l  ?4 C8 q8 s+ ^; d1 W+ Z# a; _! t  d' r
InputStreamReader ir=newInputStreamReader(process.getInputStream());  % R/ T- j9 L8 o& X0 @% k
* a6 [! r6 f+ Q5 s5 b' C. h
LineNumberReader input = new LineNumberReader (ir);  
. s1 ^# c: K7 q+ R8 g. j
" j; a; [3 m) EString line;  
; s7 l& m  y( k! w8 T# g+ q* i: q% t4 `
while ((line = input.readLine ()) != null)  ! U6 X# a% \: b- Z$ D$ [
) F, S5 U5 u8 Q! z
System.out.println(line);  0 G( }9 e, H3 x, ]% q: ?
2 u7 [7 z/ x9 M8 D) Y# _
}  
1 d" ?% N; S; f. D, {& n$ S; z9 b) ]- P- J. a1 ~2 R3 }! G7 _% ~: `
catch (java.io.IOException e){  . v3 U7 P8 U1 y' F4 Y$ S* e7 K/ c3 i

8 l1 q6 _! E4 rSystem.err.println ("IOException " + e.getMessage());  
% _7 X9 Z. W7 R/ e
, E. j5 P9 ]/ G9 [}  % J' u0 j3 W+ Y+ N# u

' C; j+ y0 m8 B. C* |: Y4 {+ ]: B4 T3 t! _9 Z/ G
问:  . q( T. \  Z# q! z1 q+ q% ^
如何产生签名applet,以使applet能够访问本地资源?  2 l# T0 z2 s3 m+ f  o
# Z, j4 o# }. I9 T) P
答:  + S0 q' L* u9 X0 J# K$ _
在jdk1.1中,可以使用javakey命令来产生公钥,私钥,证书和签名的jar文件,详细资料 请参考: http://java.sun.com/security/usingJavakey.html而java 2对签名机制做了比较大的改进,允许用户更灵活地设置安全权限.Java 2提供了三个工具:keytool,policytool和jarsigner来实现签名applet.例如,Joe编写了一个签名applet:SignedApplet.java,那么产生一个简单的签名applet的过程如下:  # }. p3 G3 V* y+ j) Q

' g5 i7 U5 i0 H3 L8 Y9 i
  E: M2 `: n/ X+ ~//产生密钥,密钥别名为joe,口令为sign12,存放在密钥库joestore中  & j/ I9 f) {  @0 A5 O

1 \2 S/ R* |) t' C- F0 o& Vkeytool -genkey -alias joe -keypass sign12 -keystore joestore  
3 e. m* p% v: U8 b
  |  o8 O' Q+ U9 }' E5 p/ a# W5 z/ E- M//将SignedApplet.class及相关文件打包成jar文件  ; k. H6 a( }# `& G
. `0 {* v" J( k# [
jar cvf SignedAppletDemo.jar  
+ J* s2 Z4 s; P# t( z$ m* e' e; r' d; ]! R) n! D
//利用keytool生成的自签名的证书产生签名applet(jar文件)  ; P) F( n# o) B( P2 a) \4 e

* i' h$ X7 |- ]4 p& _jarsigner -keystore joestore -signedjar joe.jar SignedAppletDemo.jar joe  
, Y% Q+ }' y# L' [* E: X/ {4 L
/ z$ @4 e, O: _' |: Y; K" K1 |( E//将自签名证书从keystore中输出到文件  
: u& x4 {6 }$ G0 ~& a1 v6 g( V
$ E- n. J+ ?! e& ~. ~keytool -export -keystore joestore -alias joe -file joe.cer  
) E+ R7 o! j3 n! ?
' ~4 M' _( e! Y5 i9 b2 `/ c0 i5 v7 ^
" P) j' d$ [2 {  W. n& [
而对于签名applet的接受方Susan,需要通过如下步骤来安全地执行  
% r: h7 e0 b8 y$ D! _8 `( Y: R3 W- O+ k
Joe编写的签名applet:  - z9 M: f8 m7 C- E6 Z, F

& ^6 [2 x0 b4 s3 X% ?* f//得到Joe的证书并将之读入到密钥库中susanstore中  
; r5 f9 C/ d2 E9 T7 C/ t# r' L& E* W) j* f  ~2 J
keytool -import -alias joe -file joe.cer -keystore susanstore  6 _6 E& w1 n2 @$ H+ P
. F0 ~& q7 K# b
//运行policytool产生满足Susan要求的policy文件  ( v% p* Z( S9 g2 e5 T2 B4 H

, a, n) W/ B8 tpolicytool  4 P* Y1 q  N4 B4 |

' v5 A8 O! A; `* Z! y//用appletviewer运行之,或在浏览器中安装java plugin来运行之.  
7 J6 ]7 |. g9 }9 X4 ]
7 c  P8 q% G6 b2 m. V  E5 i) b# |% C- [
5 |5 r- _* `' P" o: x4 T/ Y
关于签名applet在Java Plugin中的部署请参考以下网页:  
5 l2 m$ I% A: w" F) b, _$ L5 z
8 I3 ~) C* w% D. P' Dhttp://java.sun.com/security/signExample12/  
. a' r( U0 O0 T) u- r+ m9 N3 Z( f  l- f7 H5 a8 r. }0 e: T

! a5 }, Z; _1 _7 g  `1 h8 c& o6 C: n& w* I3 P
注:以上的例子为简单起见,使用了keytool产生的自签名证书.其实,用户也可以  7 {0 P- H7 j7 P) D& [

2 j- G( R0 _# O# T( R使用keytool -certreq向商业CA中心申请电子证书.  
1 v; p7 m8 ?" H- J7 Q, I
$ J$ Z; A  m5 X) y% u4 v, D, I/ Y1 F/ R4 `# p

& G6 O+ o: b7 }6 X4 S; b) A
+ i" W% O- Q; R: {/ F" Z问:  9 p- |3 n* d) s  `0 J
若通过ObjectOutputStream向一个文件中多次以追加方式写入object,为什么用ObjectInputStream读取这些object时会产生StreamCorruptedException?  ' A" R# E2 g; y
& y0 Y4 M# @) c; Z* L+ H" v
答:  
8 \/ J$ R3 ^+ E使用缺省的serializetion的实现时,一个ObjectOutputStream的构造和一个ObjectInputStream的构造必须一一对应.ObjectOutputStream的构造函数会向输出流中写入一个标识头,而ObjectInputStream会首先读入这个标识头.因此,多次以追加方式向一个文件中写入object时,该文件将会包含多个标识头.所以用ObjectInputStream来deserialize这个ObjectOutputStream时,将产生StreamCorruptedException.一种解决方法是可以构造一个ObjectOutputStream的子类,并覆盖writeStreamHeader()方法.被覆盖后的writeStreamHeader()方法应判断是否为首次向文件中写入object,羰?则调用super.writeStreamHeader();若否,即以追加方式写入object时,则应调用ObjectOutputStream.reset()方法.  / C* H' ~" i. X1 [/ }
4 g5 N4 [; g6 h6 K& k1 v+ w
问:  4 h  X1 P6 h0 B6 m. O
对象的序列化(serialization)类是面向流的,应如何将对象写入到随机存取文件中?  : M3 `4 X3 C0 ?
! `5 z2 o0 f& C/ Y: Q* V; \
答:  - H% \0 B& R8 h9 M
目前,没有直接的方法可以将对象写入到随机存取文件中.    K6 a% _: B3 Q* `
但是可以使用ByteArray输入/输出流作为中介,来向随机存取文件中写入或从随机存取文件中读出字节,并且可以利用字节流来创建对象输入/输出流,以用于读写对象.需要注意的是在字节流中要包含一个完整的对象,否则读写对象时将发生错误. 例如,java.io.ByteArrayOutputStream可用于获取ObjectOutputStream的字节流,从中可得到byte数组并可将之写入到随机存取文件中.相反,我们可以从随机存取文件中读出字节数组,利用它可构造ByteArrayInputStream,进而构造出ObjectInputStream,以读取对象.  2 X' E+ j* f8 a# N; P3 @. w0 C: i/ O
; I; s' l* y# o/ y
问:  * p; c2 K/ O( p$ w& A
运行RMI应用时,可不可以不手工启动名字服务rmiregistry,而是从程序中启动之?  
8 d* p' M5 P* ]0 c- l: m
  p' U' _3 ~6 `5 v5 ]! H答:  
+ Z$ m8 K- t& u/ J9 a$ C' E可以. java.rmi包中提供了类java.rmi.registry.LocateRegistry,用于获取名字服务或创建名字服务.调用LocateRegistry.createRegistry(int port)方法可以在某一特定端口创建名字服务,从而用户无需再手工启动rmiregistry.此外,LocateRegistry.getRegistry(String host,int port)方法可用于获取名字服务.  
" v' ]8 B+ H4 W7 {
3 [* Z7 N. g+ U, s3 ?" q问:  ) W  f7 @9 _! |' r) R
使用类PrintJob进行打印操作时,应如何设置打印机名等打印属性?  
+ ?! i8 \, Z% \; V7 u! x7 J
5 Y0 ~/ f. _/ \* @9 O/ g$ s答:  . [  L9 f1 `1 a& N
使用如下方法可以获得PrintJob的实例用于控制打印操作:  
9 o2 @) A, l' O. @9 m& m  h6 @# j( `

# D3 c+ y: b6 L, c* [3 fToolkit.getPrintJob(Frame f, String jobtitle, Properties prop)  2 @$ f# I1 K% F. C- _( d

4 v+ d! G; q3 I4 S那么对于打印属性的设置可以通过对prop的属性设置来实现,打印属性包括:  + r8 \& {8 s& P! i

5 ]8 j, w8 I' ?( x9 r- V1 p4 kawt.print.destination: 可以是"printer"或"file"  6 E# W# w8 [" E5 [

: a: K& P+ _* ~- ]% z9 {: c' M6 gawt.print.printer: 打印机名  
& P. v) ~  F0 U: m3 u$ Q: q: l& m6 T: e' g! m3 M5 ]7 o
awt.print.fileName: 打印文件名  $ g' e2 }& E: ^
) L, g4 A& W4 V* P9 r5 v
awt.print.numCopies: 打印份数  6 p5 w6 O& T, S* t$ Z* p6 {
* w/ \7 k+ B( e6 Y% Y- o
awt.print.options: 打印命令的打印选项  
' ~- ~/ o3 r: f* Z& v; C" J/ P: g( N% p9 M' |$ N
awt.print.orientation: 打印方向,可以是"portrait"或"landscape"  & l: t& s+ v$ d* ]& V' Y9 n

% {/ ^6 A  J. M* o8 \) ^awt.print.paperSize: 纸张大小,可以是"letter","legal","executive"或"a4"  $ I1 O5 I. ]! U" ?5 c2 a

+ }; P0 {# A. x9 b2 Z2 j! x5 @" I( b5 W( ]% z6 q  U" E; g

7 k; W4 m  f) m9 R: L' [% o  m. I' U+ S# y3 d2 n" d6 r
) s. p0 G) k1 Q. B: x% K& K4 _: X
0 _8 E* T# g1 I3 H1 l- h
问:  
. c/ g0 [+ e" r, Y; r在JDK1.1中Thread类定义了suspend()和resume()方法,但是在JDK1.2中已经过时,应使用什么方法来替代之?  
& B" a5 i% }% b+ Q- p1 Z3 ~- a* F6 p) x4 X
答:  
- U( C# ^! Z& j) O1 \1 S) K6 J3 A& z1 fThread.suspend本身易于产生死锁.如果一个目标线程对某一关键系统资源进行了加锁操作,然后该线程被suspend,那么除非该线程被resume,否则其它线程都将无法访问该系统资源.如果另外一个线程将调用resume,使该线程继续运行,而在此之前,它也需要访问这一系统资源,则将产生死锁.  0 H9 E; D1 F) t/ w1 B

' m+ d# m! k# c) d2 d8 x1 c因此,在Java 2中,比较流行的方式是定义线程的状态变量,并使目标线程轮询该状态变量,当状态为悬挂状态时,可以使用wait()方法使之处于等待状态.一旦需要该线程继续运行,其它线程会调用notify()方法来通知它.  4 W4 {+ T) ?" m5 m
2 K% Y1 _/ u: y8 P1 {/ k0 O1 @
问:  
. y# ?4 v$ V# V使用JDBC编程,应如何控制结果集ResultSet的指针,使之能够上下移动,以及移动到结果集的第一行和最后一行?  
4 N$ a. \) H: f4 z+ l- F
2 k4 S3 c& N% B- k8 L* s+ @答:  
3 M2 R! Y% G+ S+ W2 f6 i) E$ A在JDK1.1中,ResultSet类中只定义了next()方法支持数据指针的下移.但在Java 2中,ResultSet类增加了如下方法支持数据指针的移动,包括:  
" h( V/ _( `: V/ Q( B7 L! T3 B& ^+ ^# n) O9 y! v8 C( @  H
  _; z% L. n" x8 ^
ResultSet.first():将数据指针移到结果集的第一行  
  J7 G- o7 I, Q* X4 ?9 C2 H5 a8 q# A1 S, W1 z& s
ResultSet.last(): 将数据指针移到结果集的最后一行  
9 G6 ~" d. K* b$ ^; N" ~6 X
( }1 r1 k6 ?. X% f/ T4 }  xResultSet.previous(): 将数据指针上移一行  0 c# B+ z  X7 g! B) |

! P0 K9 H* o1 ^7 i- J6 ~& x6 w$ s9 C( A. l5 x
以上的方法定义在JDBC2.0的规范中,所有支持JDBC 2.0的JDBC驱动程序都可以支持上述方法.目前Intersolv和OpenLink等JDBC驱动程序厂商均有产品支持JDBC 2.0 .  # r; l0 ]9 ]3 \- }

0 ~; ~5 }' \, s6 ]  P& Y& n
5 O& n! h4 I! _, c: d$ p问:  # ~% Q+ ^/ t+ e% W
哪几种Web Server支持Servlet?如何使IIS支持Servlet?  
$ {0 K+ u0 ?9 ?" ~% Q/ k  |. S* S; N4 @/ P+ O
答:  
9 ~9 F0 ]8 U0 X目前,支持Servlet的服务器端产品主要有: Sun公司的Java WebServer,Lotus DominoGo WebServer,BEA weblogic Tengah Server,Jigsaw,NetForge,AcmeServer和Mot Bays Jetty等.  
2 ]% ?, g# O) ?! f6 a" R' X
* f! ^! ]& L6 ]8 D此外,一些第三方厂商也开发了Servlet engine,以使其它WebServer(如Netscape Web Server,IIS等)能够运行Servlet,如LiveSoftware的Jrun(http://www.livesoftware.com/ products/jrun/)等.  + x$ z( c" i+ ?, `2 V8 M( k

4 @, ^3 @5 V7 o: S' r问:  
% p( e) g0 O: `! z0 Z* M$ d$ u如何在Java应用中将图像存储到图像文件中?  0 g2 K. B5 w# q) |
& A5 \# F- @( E/ f# |1 Y8 j
答:  
1 A, \, l, j" }. D8 U% VJava Advanced Imaging API(包含在Java Media API中)允许在Java应用中执行复杂的,高性能的图像处理.JAI API提供了存储图像的能力.目前,JAI API支持以下几种图像文件格式:BMP,JEPG,PNG,PNM,TIFF.下面给出了将图像存储到BMP文件的一段代码:  
5 |0 \$ Y0 K  F% d0 Y) d- o. [  J( Y* A+ f

# F4 u& q5 i$ g: C! F# M( u4 Q  AOutputStream os = new FileOutputStream(fileToWriteTo);    Z* }& q1 o0 r$ f: W+ Q

1 m  c$ M: K1 \2 q) i2 u( OBMPEncodeParam param = new BMPEncodeParam();  1 T1 D  Q8 k2 o9 W% _+ `$ O
# Z5 b" n% T  j/ }, G3 D) }3 x
ImageEncoder enc = ImageCodec.createImageEncoder("BMP", os, param);  : Z/ Y' B0 P" |" A. D+ k4 l* S
7 A' f+ x/ S, ~: a% V3 A2 x: e
enc.encode(img);  
: {% Q/ S  V4 Q5 v' R$ m5 D& s* M# X  G) _6 E6 \0 `! c+ Y
os.close();  
5 T5 x& r8 C# T
: s4 u" ~* m' N% P+ \有关存储图像文件的编程指南请参考以下网页:  2 \) P- P! `& h6 A( K

0 e, a& `8 ]; ]3 w0 Lhttp://java.sun.com/products/java-media/jai/forDevelopers/jai-guide/  $ x5 u) n* D( V

- j3 t6 @" c* D) T  _: ]3 N% f7 J2 _6 z7 ~
! J8 ~- s% y& a  J3 A% U

& G6 _/ [, `" X+ {% u5 H; C# Q问:  5 z) X; @" y1 u4 u
如何用Java语言向串口读写数据? font>  % h4 P/ s, i7 b7 `( s3 G4 n# h

6 X5 ^+ n$ H7 x/ |答:  / Y: X$ ?7 \0 h4 I+ y
Sun公司的Java Communication API2.0可用于读写串口,它支持RS232串口和IEEE 1284 并口,提供了一种与平台无关的串/并口通信机制.  
6 i/ k$ d" g( q, s) n  z* i, c' z' b7 Q8 H
详细文档,请访问:<a href="http://java.sun.com/products/javacomm/" target="_blank" >http://java.sun.com/products/javacomm/</A>
作者: wangyu249    时间: 2006-11-21 15:02
<p>&nbsp; hehe</p>
作者: wl820609    时间: 2006-11-29 07:21
<p>顶!!!有最新的没/</p><p></p>
作者: deven1985    时间: 2010-1-6 00:44
楼主辛苦了,现在先顶下,以后慢慢看!




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5