QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 4912|回复: 3
打印 上一主题 下一主题

[转帖]Java常见问题集锦(来自Sun中国官方站)

[复制链接]
字体大小: 正常 放大
god        

206

主题

2

听众

882

积分

升级  70.5%

该用户从未签到

新人进步奖

跳转到指定楼层
1#
发表于 2005-3-31 01:31 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
问: 如何设置Java 2(JDK1.2)的环境变量?  
! s# C" B/ ], V; s1 T
! a0 V- n6 ~% }4 W) C答: Java 2安装后,需要设置PATH和JAVA_HOME环境变量.与JDK1.1不同的是:设置好JAVA_HOME环境变量后,JVM将自动搜索系统类库以及用户的当前路径.  ! S, J: b, F7 B0 u, S, P" \( |
Java 2环境变量的设置如下例所示:  
2 J" l8 ]+ z. e5 v0 sSolaris平台: setenv JAVA_HOME Java2的安装路径  
( h0 e, P  v  k; b. n, Psetenv PATH $JAVA_HOME/bin{PATH}  
! G# ^* R/ w: [5 D/ O( E% S3 hWindows平台: set JAVA_HOME=Java2的安装路径  3 o. D7 h2 @7 O4 E& Q
set PATH=$JAVA_HOMEbin;%PATH%  
6 d: F. n+ g. f% e1 J6 s* k5 Y8 f4 C2 y
问: 哪些Java集成开发工具支持Java 2?  
" ^, z. l5 m# \. }2 `8 o6 J( P/ E4 E2 Y8 U3 m# l
答: 目前流行的Java集成开发环境,如Inprise的JBuilder,Symantec的Visual Cafe, Sybase的PowerJ,都支持Java 2.  9 |" d# R9 m% U% d
8 |! K2 t6 {5 J, ]
问: 如果在Netscape或IE浏览器中运行Java applet时出现了错误,如何确定错误范围?  
; z7 P1 j2 `0 D( B
4 D8 W+ L5 ?6 A" f! R( {答: 当java applet在浏览器中运行时,使用的是浏览器本身的缺省JVM.而不同浏览器对JDK的支持程度也不尽相同. 因此,在Netscape或IE浏览器中运行Java applet出现了错误,建议使用JDK提供的工具appletviewer或Sun公司的Hotjava浏览器来测试该applet,以确定错误的产生是与浏览器相关.  
) }: r- U& j# s, I/ r    如果applet在appletviewer或Hotjava中运行一切正常,则错误的产生是由于浏览 器不完全兼容JDK而引起的. 此时,解决方法可以是使用Hotjava浏览器或者安装 Sun公司的Java Plugin.  
0 l  L+ m# v+ g/ _    如果applet在Hotjava浏览器或appletviewer中运行即发生错误,则应当根据错误 提示检查applet程序.  
- M! U" p6 a+ u2 x/ @
( {' g2 L$ P/ x6 U2 \% g: T) S问: 当用JDBC向数据库中插入数据或从数据库中提取数据时,为何有时中文字符会显示为乱码?  
, M$ t% f" d! `' ?5 w! H) I8 T" U; L
答: 这个问题的实现通常与各个JDBC driver的实现有关. 目前大多数JDBC driver采用本地编码格式来传输中文字符,例如中文字符"0x4175"会被转成"0x41"和"0x75"进行传输. 因此我们需要对JDBC driver返回的字符以及要发给JDBC driver的字符进行转换.  
6 w' d- I) A( [* k
, j  J; S' F2 a3 A) H5 ^当用JDBC driver向数据库中插入数据时,需要先将Unicode转成native code; 当 JDBC driver从数据库中查询数据时,则需要将native code转换成Unicode. 下面给出了这两种转换的实现:  / _0 _( `$ Q- e& O3 ^

$ @$ A9 P* l2 W; Q
( ^" O- y3 E0 ]/ u) `String native2Unicode(String s) {  ! ?# |: D( }. P1 V$ B$ X3 a

$ e  P0 Z" y! [if (s == null || s.length() == 0) {    _. a# S& j3 d' q
1 X+ Z5 \. P2 {' h
return null;  " f2 B5 ~! j: p3 H) m. f

2 {6 R" m3 b- x4 V9 \1 {! {}  ! I7 N' b5 ^) q9 O; C, ^
5 U/ e  v+ l- j2 \' a0 T
byte[] buffer = new byte[s.length()];  
+ i( {' R* E1 n: q3 M6 U% a5 C; b9 H
for (int i = 0; i s.length(); i++) { if (s.charAt(i)>= 0x100) {  
* ]- ?! j, O! ~$ v2 \! z2 e. Q! ^* H$ i3 O% w/ F% I9 Q$ v+ L5 a5 A
c = s.charAt(i);  , L% I- Q3 Z5 T, L

! |3 P# ]0 a( V" F/ L4 ^1 X  Jbyte []buf = (""+c).getBytes();  
$ u% W' H; \  F% F& h8 U  e
3 o6 I9 l2 c5 G* bbuffer[j++] = (char)buf[0];  4 Z3 T( ?, ^) k1 D5 G
4 s2 J9 t4 T. Q* W/ |9 p8 @) z
buffer[j++] = (char)buf[1];  
, K8 {& F9 \  D! Z. n
0 S! B- t# F" A) D, M}  . y% }% q7 V0 D9 ^# d- `

1 n2 S3 h6 E' P5 |  {: kelse {  8 `- x) G8 i1 V) J4 H* d
$ B* S, ^! O' A1 g/ H
buffer[j++] = s.charAt(i);  $ w( M- ~/ S; N! c! U* B/ `
/ |( T7 z& S! E
}  5 C; X1 E) V4 J+ h

/ v/ O! i2 Y. y4 M! s}  : l$ A4 G' w7 F, _5 I5 M* K
6 ]7 e: i& }$ B; [6 X' B
return new String(buffer, 0, j);  
7 K3 s$ a! U" C1 E1 m5 ^' b: E1 J1 U4 M& d
}  
5 ^; c4 O) _9 ~7 V0 H5 n% R1 z4 r" z: q
除使用以上两个方法之外,有些JDBC driver如果对jdbc driver Manager设置了正确 的字符集属性,以上2个方法就不需要了.  
$ O: B, D0 U( h/ @- R$ n$ Y4 R2 c) }6 f) T9 ^9 N/ A
; f* F: Z( f6 `% r/ g; u
问:  
* \- T3 Y3 E) [; c当用Servlet来处理http请求并产生返回的HTML页面时,如何使HTML页面中的中文字符能够正常显示?  8 u. v9 Y, C/ b

* Q8 ^0 j: k7 N" C- b3 _答:  
3 W) }; _$ v; Y6 vjavax.servlet.http.HttpResponse类用于产生返回页面.通过HttpResponse定义的方法getOutputStream()可以获得ServletOutputStream的实例,这样用户就可以利用ServletOutputStream.write方法向输出流中写入返回页面的内容. 但是ServletOutputStream使用的是缺省的编码方式,如果要使返回页面中的中文字 符能够正常显示,最好显示地指定所用的字符编码方式. 通常需要构造一个 OutputStreamWriter , 例程如下:  
. C7 p/ T7 ]! r( T8 ~
! {2 }, [! @7 A1 z2 K
; O, J) f6 T5 A% q) ^public void doGet (HttpServletRequest req, HttpServletResponse res)  
0 I# G" g5 M1 l! w: ?. ]! }$ Z. Z2 G/ l3 J' I# ^: T
throws ServletException, IOException  
" d7 p; K  R0 C' {. F& ^6 l. L" H
) X! r& _' P1 C8 M) r4 v- ]{  6 I, l6 {  t- L: y! P# n( J  ~

$ y) U0 P6 h2 l, j$ L" rres.setContentType("text/html");  
3 F2 g2 v* F2 c& R! `$ C! U' I2 E6 F- o% e
ServletOutputStream out = res.getOutputStream();  + \, ?$ T" A* [. w/ v) |3 o$ U

- \. h% H/ j' d: S4 ROutputStreamWriter ow = new OutputStreamWriter(out,"GB2312");  
9 l3 D) b9 Y6 K
3 q8 j( d7 |" w# i8 now.write("这是测试");  ) Q  n4 ^( N' ^# h
7 p  X* S% N9 w* a* }' w% Q
ow.flush();  # r- ?- c* k! K
- K+ p9 g* U& I8 i, v2 Q4 Y
ow.close();  " n! f" R& X0 x" j
) F4 Q- Q, l; b6 Q& @( O$ Q
}  
& z2 D! {/ ?; n4 Q, w; M
8 X' Q, }) d, R+ r/ `( G( V' ~, h* X+ [
问:  2 E, e3 @+ R8 f" x4 D
如何设置Java WebServer的CLASSPATH,以包含用户的class文件?  
: h: c" O, k& Z+ W
; W, {3 o& H: {答:  
2 C" @* x# r/ A有两种方法可以设置Java WebServer的CLASSPATH环境变量,以使用户编写的Servlet能够调用用户的class文件.  . [8 {1 F1 W4 E+ Y/ @
将用户的class文件放到 JavaWebServer_Dir/classes目录下,在Java WebServer 启动时,classes目录被自动加入到CLASSPATH环境变量中了.  $ M' i' e9 e2 T8 f0 i+ U
修改httpd.nojre文件,将用户class文件所在的路径名加到CLASSPATH环境变量中.  
5 i: G8 n; ]  a, r7 |& |  H
4 K; N, W6 y; O. X. }  r9 M/ W$ @! g, B
问:  ( N: h6 x, u; p8 e& S% ^1 r! Q
为什么在Windows平台上用Naming.lookup来获取远程RMI对象时会很慢?  6 y8 k( k2 |" b. c9 Z. O! ~

# X2 p" Q7 f% Q6 Y$ s答:  " R% ~, @* W/ `! V: n9 b3 _" l
机器的网络设置不正确很可能会引起该问题的发生.    t) r/ I  l2 p  D7 @: u
RMI使用了Java网络类,特别是java.net.InetAddress类,它将查询TCP/IP的主机名, 包括IP地址到主机名的映射和主机名到IP地址的映射.在Windows平台,这种查询功能 是由本地的Windows Socket库来实现的. 因此延时是发生在Windows库中,而非RMI中.  
3 P6 F) E" K7 V0 R: N7 [
* t  Z/ t# P& _如果你的机器设置成使用DNS,问题通常是DNS服务器查不到主机名,你所发现的延时 是DNS查询的延时. 请尝试将RMI通信中涉及到的所有主机名/IP地址加到本地文件 winntsystem32driversetchosts或windowshosts中. 格式如下:  
) ~1 n7 j$ p8 z2 S0 m* b
) T. p: O0 h' x0 `+ V9 ^IP地址 主机名  ' g% q. p# b7 G& D8 N6 B& \1 z

" \8 E: g7 K4 v# l3 ]; R& b: ]如此设置应当可以明显地减少查询所花的时间.  7 X. \2 Z  d; N3 l! s
3 j' D1 B: Z8 P. S
问: 编写Java application时,如何设置proxy的信息,以便访问外部网站?  2 C* d5 e, D2 _# g
3 g7 M* m, S' Y* d( H+ U
答:  
7 U6 ~6 L3 u/ Y6 x& U若在java application中访问外部网站,首先应设置proxy信息,样例代码如下:  
' D% ?' f$ K8 z$ c7 j- C
# t+ P  K  `1 i. O4 m! m3 h) R
4 F- Y0 X% O. L! ?import java.util.properties;  & Z( _# n' |" i( Z/ \' Q5 N
' n5 X7 F: C* Y+ E# B& O# H( [
.....  , Y* _1 c- o$ u8 C1 o* i

% O4 E% A% T6 ?$ v9 HProperties sys = System.getProperties();  
  F3 h# [; B4 h5 |( @0 i! z
0 u5 ^& Y. J% ?9 Hsys.put("proxySet","true");  
' K6 J- Z/ V7 }& g* s
% k# a9 W& f* {0 A1 O4 rsys.put("proxyHost","myHTTP.proxyserver.com");  & p7 H: H9 R0 K2 D! z
- A! U- b! p! Q% J
sys.put("proxyPort","80");  
# o7 B; \. [- b4 d
# f7 @/ \/ B4 V0 c' k1 o) m6 iSystem.setProperties(sys);  3 o- J6 n% X+ A

8 q6 F9 L: w- k: ~2 b1 l
: E& d3 |, a% W0 Z! D, P- \- `' g# M- I/ S! y8 R
u = new URL(website);  % ^; S* g+ D$ Q6 H

3 [" Q2 Q4 [/ A, G6 g0 N8 Sconnect = (HttpURLConnection)u.openConnection();  $ [' l) _% l0 u' K' Z& h0 x
7 v1 m! x4 ~  J9 b- L' `
.....  
, X3 T$ w- k8 K1 T: j) w% h
6 t/ C* C; C  w. S" v# A* P问: Swing组件JList的列表数据修改了,如何通知JList改变显示?  
6 o) U: D1 k  w. F
) ^2 p; Q" s( N! m) ]答:  
  {8 e' y8 j, WJList组件有一个单独的显示模式ListModel来表示JList的显示数据.  
& \' q0 x9 d- Z6 k3 aJList创建以后,JList数据元素的值及数据元素的数量可以动态地改变.  
% o. s7 b+ ]8 J" K. f: B2 X7 OJList在它的数据模式ListModel中观察数据的改变.因此,一个ListModel 的正确实现应当在每次数据发生改变时,通知事件的监听者.  * G/ e& u/ s* O3 T
当使用构造函数JList(Object[])创建一个JList的实例时,系统将自动 创建一个DefaultListModel的实例来存储JList的显示数据, 可以调用 DefaultListModel中定义的简便方法来动态地修改JList的数据,如 removeElementAt(index),addElement(Object)等. DefaultListModel 在修改数据的同时,将通知JList关于数据的改变.  
. C- U  U; ^" n/ Y' `' f9 S' ]4 x4 ]' `/ p# Q7 t9 ]7 t+ W
问:  ( t) C' `3 |# V; T$ B
在Java applet中如何实现一个模式对话框?  
9 }% E6 N1 q3 Q6 a5 x& |  N2 e5 J8 m: P2 B, h0 K! g
答:  
7 I0 p( T+ g1 B: T' F- _/ l- w在Java applet中实现模式对话框的关键就是在创建一个对话框的时候 要为该对话框指定一个正确的父窗口.因为Applet是Panel类的子类,不 可以作为对话框的父窗口,所以首先要获得applet所在的窗口,作为模式 对话框的父窗口. 样例代码如下:  
% a; q) }0 I5 w' Q$ `: t* i8 N: O! @& ~

% w: `: Z* B: h; ?( O+ f  `- K: J.....  
2 w7 R( T" q2 u& J/ Y( Y- c
0 d, [% o! B/ s+ uDialog d = new Dialog( getParentWindow(comp),title);  : E! Q3 q$ m; ?1 q

* {/ @3 H5 u- U% h4 G. ?// comp为applet上的任意一个组件  , m# m: a/ g) I; T$ q

2 a4 @& W% Q  Y/ \5 @....  6 |* O* O5 t' ?( t" C0 A7 n
1 K, X: r( N+ N

! t$ T1 g2 _; ?. _
8 A6 t  {9 v7 A& ~; p/ Mpublic void getParentWindow(Component compOnApplet,String title){  
" u* t8 e+ h1 ?+ g1 {  p, H5 ]8 g2 ]6 H
Container c = compOnApplet.getParent();  9 g6 F$ ?6 G( O% m: U
3 h  b& B! ?; O5 |% [7 i1 Q
while (c != null) {  8 g3 U  b5 W9 C1 Z, F  {/ @

$ \) a$ z; E' x/ {if (c instanceof Frame)  0 ~2 |( `. @& d  m, z8 u
, _( `3 j- [7 m7 r* l
return (Frame) c;  : y/ l3 L) l% q' G' K- W* S
8 F! N$ X7 e+ P2 [3 P/ E
c = c.getParent();  0 |) Y; Y' e* [. v

; u; z  Y& a) T6 C& w}  - S- D, L" \$ w  v9 L( E
1 @! |4 z$ m: s2 A/ q
return null;  
, ?* _3 }' ^; H$ a+ A& u1 A3 G7 g& M  Q3 i! v4 ~
}  # S5 _" X2 h" i& {/ c7 Z. t
4 c% m9 E& y( V% i5 i3 w
问: 在Java applet中如何显示另外一个HTML页面?  + U1 F+ I, I; i, R; m$ ]0 A. i

9 L0 V1 g# P' M/ D2 j答:  
2 @6 `, W; Z) G6 g6 B通过java.applet.Applet.getAppletContext()方法可以获得与该applet相关的AppletContext, AppletContext.showDocument(URL)方法就可以使applet所在的浏览器显示另外一个网页.  # [  U  r& s: K9 r, A6 r, w) `
) `! X- |* K8 y7 A' m$ x- b! G3 m
问:  
+ M# U* ]# X& D用JDK实现的签名applet,可否在Netscape或IE中运行?  ( K# |6 Z$ Q% z/ J3 W! X, @
  q! C8 l( Z6 i
答:  
7 e. r# l- s# w- L- I用JDK实现的签名applet,不可以在Netscape或IE中运行,但是可以在Hotjava浏览器中运行.  ' |% P! C" r3 c  J' n3 z& i

5 F- O) u1 g7 I% e# c1 C) ?4 O不同的浏览器提供了不同的签名applet机制,如Netscape提供了zigbert工具和 Capability API, 而IE则需要使用CAB文件. 但是,无论是Netscape工具产生的 签名applet,还是用IE产生的签名applet,都不可以在其它的浏览器中运行.  4 t7 T* D7 a# ?0 k* {: E

9 t- E  _" e- Y( m  K3 c/ z& B如果要使JDK产生的签名applet能够在Netscape或IE中运行,解决方法是在 Netscape或IE中安装Java Plugin,则用JDK实现的签名applet就可以在这两种 浏览器中运行.  8 ~: K7 w- @& t8 e

" C  J& `' R' P问:  
/ a( m' e; B0 X# K; f$ g) f& d用JNI技术可以从Java应用中调用C程序库,但是如何使该C程序库可以调用另外的C程序库?  
0 \2 r/ _6 ^, s
. ]# \7 H6 D$ t2 a% z答:  
6 U/ {+ ]% i% m: o9 b9 W如果一个被Java调用的C程序库C1仍需要调用另外一个C程序库C2,那么在编译C1的时候应当联接程序库C2,步骤如下(Solaris平台):  
* C+ S$ p, N- D1 c  i' L- G编写调用C库的Java文件,并编译.  2 L5 u: ?+ u1 c2 u) [1 H
javac java文件名  * {* x! Q" k) R% T1 `
; O/ C. N' v6 I; c' B( S
7 _" j) n& t7 _7 E
产生C程序头文件  3 R+ i8 W7 B! S( ^8 ?
javah -jni java文件名(不带后缀.java)  + @; d" j8 k1 \% ?% u0 d
3 D$ A. z, w' F3 _8 E' f( n2 U
' b7 G) b7 f3 N5 Y! i
编写被Java调用的C程序C1.c,以及被C1调用的C2.c,并编译.  ' f5 Q3 B3 w9 i0 E2 O
cc -G -Iinclude路径名 C2.c -o libC2.so  2 P- ~2 I# \( I' K( f. k: E
cc -G -Iinclude路径名 -lC2 C1.c -o libC1.so  
. I3 b/ z: f: W; {: q1 I, P- {8 ^% q+ Q
4 Y  x( V1 V5 `1 Y, c4 E
设置环境变量  
, y2 J. f. Y/ |setenv LD_LIBRARY_PATH libC1.so,libC2.so所在路径  ) `% }/ G* `: ]( b/ }* ]  l
{LD_LIBRARY_PATH}  # R& z% [/ A. ~: b

4 U( H; O# m! D1 `$ z" P* A6 x4 l7 D9 i/ \* p* i+ D- f9 h3 N
运行java应用  
& p! j6 b( I' B" y; h+ G
4 }5 k" J( Q/ ]/ c6 I9 l1 R/ k1 Y+ v2 ?6 |- y% {6 D
问:  
: _8 p& i- T. D4 S& [0 ~2 M3 k在Java语言中,如何列出PC机文件系统中的所有驱动器名?  3 S; {0 b  W$ c. `) I
/ f5 @* s9 \  f- S+ \1 v! e
答:  7 [: {- J( z3 T' }  ?6 i1 ^
在Java 2版本中,java.io包中的File类新增加了方法listRoots()可以实现这一功能.  
! N  V. F: V& O7 W" x& h5 j
6 \# Q/ q2 Q- @) z. p& _. _# I( V4 C问:  # S( `, k# a& B/ q8 m4 s% m/ X, |
为什么Runtime.exec("ls")没有任何输出?  
. e- _) T  g+ J$ i) p- p8 `. t$ P2 ^1 V8 o
答:  ! B6 o8 C. D% L0 b  L! V, i
调用Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息. 由于调用Runtime.exec方法所创建的子进程没有自己的终端或控制台,因此该子进程的标准IO(如stdin,stdou,stderr)都通过Process.getOutputStream(),Process.getInputStream(), Process.getErrorStream()方法重定向给它的父进程了.用户需要用这些stream来向 子进程输入数据或获取子进程的输出. 所以正确执行Runtime.exec("ls")的例程如下:  
2 T& ]( Y# u2 l% \2 s' F' t, g+ G8 [
# L) S/ q/ ^( b  W6 `
try  ) J4 Z! r% C' K& o$ q

# s% I# m7 p! s4 D8 E{  / t. I) {; k) P  s- G) @$ ~

7 Y4 o: y. E, I/ k# x" j3 ~process = Runtime.getRuntime().exec (command);  
% y( h: \2 u5 h5 Y+ t
2 I* `& G8 `  q1 Z  A3 ?. OInputStreamReader ir=newInputStreamReader(process.getInputStream());  / h, M: d/ k' [: d9 I
% P/ ^( M* D4 x/ I
LineNumberReader input = new LineNumberReader (ir);  
7 s* H  ^3 d& E, ^2 T. ]( e3 Z) s8 ]; L. r. j% e2 B7 C+ c
String line;  # F8 u0 R7 ~$ q6 J9 O& M
6 H0 u, U7 {! j; l5 q; k$ U; M8 D
while ((line = input.readLine ()) != null)  # I2 N+ ]) t" F# u, t: K9 n

5 Q" L' [" G" _System.out.println(line);  & n( ]8 a3 t6 x: ~9 F" i$ R
  c! J  A  w' c9 h6 z
}  8 V5 q0 E; E2 b# ]1 M1 d

) j5 J" b0 _9 `4 @7 k. l  Qcatch (java.io.IOException e){  ! t1 m+ c! g- H+ ~
: }# N! j5 m1 G
System.err.println ("IOException " + e.getMessage());  
+ Z- ~' }' J# A  u" F; w! @1 V' s% ^7 p9 r0 R
}  
. D( u  n& q) G/ b4 K
8 l9 I; d$ O+ \6 V' C$ l9 P) m% S. T4 p. v- S
问:  4 I8 T- T+ v1 |4 K+ ~7 m5 C
如何产生签名applet,以使applet能够访问本地资源?  5 U: _! N; Z) _$ `

, r2 i% {4 N* F7 f( \  F% q# p答:  + ]2 S; W8 e, 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的过程如下:  
* Y# U1 R+ Q; k6 `: ~3 T" h- t% e% m# o9 g
' e1 [2 t3 [1 B9 c# @8 U
//产生密钥,密钥别名为joe,口令为sign12,存放在密钥库joestore中  4 H7 s+ e. [1 X) D5 L

" S( i  n# |0 q2 y7 bkeytool -genkey -alias joe -keypass sign12 -keystore joestore  8 x3 Z7 O4 b$ g6 K

7 `! s  g5 s/ I) i//将SignedApplet.class及相关文件打包成jar文件  
8 m7 l4 M9 q. Y! {' n7 T' o; w, `  N# n# F. T
jar cvf SignedAppletDemo.jar  & L& N7 g& i1 q; V: S; s* f) E4 ~

" L( v( N3 |2 `. {2 ^3 C//利用keytool生成的自签名的证书产生签名applet(jar文件)  ; ~8 f  B6 z, j# D
: {/ K$ o/ I5 G8 R) ?2 d, a! h
jarsigner -keystore joestore -signedjar joe.jar SignedAppletDemo.jar joe  
0 `- S; L' g8 A6 q5 h+ p" q, y1 e! e( `# C1 i
//将自签名证书从keystore中输出到文件  
0 F% m8 N8 a5 G: d$ a  V# s$ i! Y/ v9 {, N3 f
keytool -export -keystore joestore -alias joe -file joe.cer  
9 j; O9 E8 @; W2 H' T4 }6 D/ `  H4 p: j9 d! }, O# }% F, D

  r) d" [7 c8 s1 t+ f
2 {. H' e0 v, I0 O# H1 g& g6 F而对于签名applet的接受方Susan,需要通过如下步骤来安全地执行  
& y6 X. i" U( Y+ Q
* b5 P% [  u0 y( C0 S! ^7 F  TJoe编写的签名applet:  
& u1 e: z1 s9 w. s, n+ C4 ]3 ~( t+ o. m; E! ?/ J
//得到Joe的证书并将之读入到密钥库中susanstore中  
8 u4 {" W7 z( E% j2 u- K, F
. c5 q2 Y# w" U( t/ G) Ekeytool -import -alias joe -file joe.cer -keystore susanstore  
; \! Y3 y# s. q% _& d* U. ^
/ G  Y1 B) ]5 ^3 S# G5 B4 u1 a//运行policytool产生满足Susan要求的policy文件  
* p* v7 Q. l/ @# N2 x
  ]. R2 A4 S+ i6 c# ]( `policytool  
1 q  I: x4 U( ^+ A/ q  ?4 p; g& \; M0 Q9 ~
//用appletviewer运行之,或在浏览器中安装java plugin来运行之.  
* N9 B1 D' M! e5 P5 ?  V1 Y$ F( g) k3 x+ |

2 M; H% t7 Z0 @& P, j- j8 H7 z& y2 q1 D9 |& Z0 Q2 Y; _1 f
关于签名applet在Java Plugin中的部署请参考以下网页:  2 v1 Z2 d% {) j% m2 |# v" \

7 |% w: y( B6 P- c) jhttp://java.sun.com/security/signExample12/  ; b5 j  R* }: x+ v
& x2 S# N- K) l6 @5 h* a

; h' H( y- D6 ^; j' v5 J; j# }8 V; i' y
注:以上的例子为简单起见,使用了keytool产生的自签名证书.其实,用户也可以  
% R* A& i) j! c/ e$ h
/ R  J# z; P- s5 E% s% X使用keytool -certreq向商业CA中心申请电子证书.  ) q1 C- J$ u/ r( E3 k

, e6 |- V9 L; C# G6 I  i# P  S  u" q! T; D( N6 m
0 h4 w- i; f9 L( l  ~$ a* {0 d

$ e6 Y8 B3 D2 D! ~7 }  H" Q问:  3 a, t% ]# J' N' d
若通过ObjectOutputStream向一个文件中多次以追加方式写入object,为什么用ObjectInputStream读取这些object时会产生StreamCorruptedException?  
* y  M5 j4 L; ?. Y2 b4 C. b' N# {! d) p  i- H
答:  
  z+ J/ C" U& q- q0 Q使用缺省的serializetion的实现时,一个ObjectOutputStream的构造和一个ObjectInputStream的构造必须一一对应.ObjectOutputStream的构造函数会向输出流中写入一个标识头,而ObjectInputStream会首先读入这个标识头.因此,多次以追加方式向一个文件中写入object时,该文件将会包含多个标识头.所以用ObjectInputStream来deserialize这个ObjectOutputStream时,将产生StreamCorruptedException.一种解决方法是可以构造一个ObjectOutputStream的子类,并覆盖writeStreamHeader()方法.被覆盖后的writeStreamHeader()方法应判断是否为首次向文件中写入object,羰?则调用super.writeStreamHeader();若否,即以追加方式写入object时,则应调用ObjectOutputStream.reset()方法.  
  T% W' f) S7 S% b7 w/ b9 C
" J* b: X1 c* Y问:  ) N* k8 l* w3 ~0 g9 M/ d  D9 C
对象的序列化(serialization)类是面向流的,应如何将对象写入到随机存取文件中?  : K/ a: W8 n+ J$ j7 D* k( L, b" C
6 T$ }/ l3 q4 Y% H: l7 [
答:  7 ]/ Q8 D# d9 X. m  o9 q
目前,没有直接的方法可以将对象写入到随机存取文件中.  0 k. P  ^9 ^6 Q1 G9 \- Q
但是可以使用ByteArray输入/输出流作为中介,来向随机存取文件中写入或从随机存取文件中读出字节,并且可以利用字节流来创建对象输入/输出流,以用于读写对象.需要注意的是在字节流中要包含一个完整的对象,否则读写对象时将发生错误. 例如,java.io.ByteArrayOutputStream可用于获取ObjectOutputStream的字节流,从中可得到byte数组并可将之写入到随机存取文件中.相反,我们可以从随机存取文件中读出字节数组,利用它可构造ByteArrayInputStream,进而构造出ObjectInputStream,以读取对象.  
2 F- M8 k5 j4 Y: I6 t2 i( A' K/ q) I- ~- c5 v
问:  
3 `9 t: i7 d' o. B运行RMI应用时,可不可以不手工启动名字服务rmiregistry,而是从程序中启动之?  
) n" b5 ]  y* S+ _) z% V/ p3 S) M7 g+ @. M
答:  
/ a9 S6 J  ]7 N可以. java.rmi包中提供了类java.rmi.registry.LocateRegistry,用于获取名字服务或创建名字服务.调用LocateRegistry.createRegistry(int port)方法可以在某一特定端口创建名字服务,从而用户无需再手工启动rmiregistry.此外,LocateRegistry.getRegistry(String host,int port)方法可用于获取名字服务.  - I# @$ o1 q. K. G
. V& i  P0 _/ h( e6 w, t9 j
问:  
2 b, P3 h$ h4 L: A& k4 B! D使用类PrintJob进行打印操作时,应如何设置打印机名等打印属性?  
" D. f* c" i6 ^, z6 |
- ~( d8 Y1 W$ D/ i% w答:  
% I/ N+ f% B7 P9 S/ K2 B使用如下方法可以获得PrintJob的实例用于控制打印操作:  
2 J# L9 K; N2 L% N. a1 H+ K% E" H
9 I" I! v6 t8 N) @) \$ U* p, Z4 l. t6 [
Toolkit.getPrintJob(Frame f, String jobtitle, Properties prop)  
( [5 ^' K" W% ^$ k; e' L1 ^  Z+ a0 ]# P: a
那么对于打印属性的设置可以通过对prop的属性设置来实现,打印属性包括:  
7 v% m1 W! v+ {9 p. l* i
6 Q- J$ _' \& z! B  Gawt.print.destination: 可以是"printer"或"file"  3 |+ k: a3 B' W" T5 Q$ T
! d4 p7 z0 D% M0 g" c& x  F
awt.print.printer: 打印机名  
! }- L# Y+ N: Q5 @6 j- U+ W, Z8 K  ~/ ~2 D9 S: i- ]6 ~4 e! D
awt.print.fileName: 打印文件名  2 y" E$ C# e/ @# y& u; B; J+ C" j

, Z; G1 l/ J4 aawt.print.numCopies: 打印份数  4 C( @, l; ^) x3 j0 p) x8 Z0 O
) T; [/ r% I5 k% W( Z5 |
awt.print.options: 打印命令的打印选项  
; o  ]' |; z1 ?# x  D, y
& Z! O1 a3 @4 r; _# sawt.print.orientation: 打印方向,可以是"portrait"或"landscape"  6 F+ }7 u2 v4 @2 T5 |- ]+ l
. b5 O4 ~& D, [' s
awt.print.paperSize: 纸张大小,可以是"letter","legal","executive"或"a4"  
2 q( i$ {- b% J4 F- w9 g0 X3 E& |8 w3 }, w$ C7 I* p, [4 S" `
; H* u7 z- [6 t% c

- r0 j- z1 \4 |4 M& |" }0 R
8 X' @7 G6 x( N6 C
6 ?" k1 j; q, U7 h# A5 }
5 N( H( [# P/ p* O& w2 ]# M问:  1 L+ W5 U/ {3 L) d( _/ J6 o
在JDK1.1中Thread类定义了suspend()和resume()方法,但是在JDK1.2中已经过时,应使用什么方法来替代之?  
0 A' j1 H; _+ \1 W: F2 v& f
: p# r6 ?/ {1 g答:  6 F- r5 `! Y5 y; i( Y
Thread.suspend本身易于产生死锁.如果一个目标线程对某一关键系统资源进行了加锁操作,然后该线程被suspend,那么除非该线程被resume,否则其它线程都将无法访问该系统资源.如果另外一个线程将调用resume,使该线程继续运行,而在此之前,它也需要访问这一系统资源,则将产生死锁.  7 A$ V- b) p5 q, F

% H9 L, J, h- O7 t- j因此,在Java 2中,比较流行的方式是定义线程的状态变量,并使目标线程轮询该状态变量,当状态为悬挂状态时,可以使用wait()方法使之处于等待状态.一旦需要该线程继续运行,其它线程会调用notify()方法来通知它.  3 @+ ^$ J" [7 Q9 `/ ~( [
* C4 @4 Z( N0 _, m( w( l9 j7 Q
问:  - X! t$ c. L; P6 ~
使用JDBC编程,应如何控制结果集ResultSet的指针,使之能够上下移动,以及移动到结果集的第一行和最后一行?  
/ e2 ?( d3 D- p
2 @' B$ z! \! Y5 F9 H2 B7 h答:  ; l$ Z5 h9 U8 U9 Q* q# n/ q8 W
在JDK1.1中,ResultSet类中只定义了next()方法支持数据指针的下移.但在Java 2中,ResultSet类增加了如下方法支持数据指针的移动,包括:  & u6 B* v! B% t5 S" H7 X) m
0 f: C% b4 [- F! Z
) Q  `1 l' x: d& t; W* ]5 x3 b
ResultSet.first():将数据指针移到结果集的第一行  
# B1 h# S; f3 q0 m' \; `! w1 }# O  E) {! C, A, H7 P' d& j* |
ResultSet.last(): 将数据指针移到结果集的最后一行  
, r+ C1 D; i( F/ T8 d. f8 a! P8 G
* l( x3 u- o/ k* Q9 ~ResultSet.previous(): 将数据指针上移一行  
3 y7 K4 C5 A; b9 V1 s8 |
  p" Y9 T! r3 c$ _6 o; n, ^' w8 U. ~: j8 t# C9 \
以上的方法定义在JDBC2.0的规范中,所有支持JDBC 2.0的JDBC驱动程序都可以支持上述方法.目前Intersolv和OpenLink等JDBC驱动程序厂商均有产品支持JDBC 2.0 .  0 P( j/ N. u" w$ R

) i; z" u+ R8 I( O5 `  P; u4 u
0 R) _; c' [" U3 O/ N4 ^问:    n. M- q3 ?% u+ h  [
哪几种Web Server支持Servlet?如何使IIS支持Servlet?  
8 E9 U/ r& c/ U; K. }1 r0 g
1 F6 H6 Z! K* i/ U* b/ t; ~答:  $ G% J! P  j$ W9 S+ P+ O) w8 T
目前,支持Servlet的服务器端产品主要有: Sun公司的Java WebServer,Lotus DominoGo WebServer,BEA weblogic Tengah Server,Jigsaw,NetForge,AcmeServer和Mot Bays Jetty等.  
& A; _  b; c6 v" k; L  E5 X
4 H* D- d* {5 z. a此外,一些第三方厂商也开发了Servlet engine,以使其它WebServer(如Netscape Web Server,IIS等)能够运行Servlet,如LiveSoftware的Jrun(http://www.livesoftware.com/ products/jrun/)等.  / m$ b6 h/ h- z  O
, p( R/ Y/ L+ f$ K& I# @: e0 W
问:  
  p" E% M5 o" b+ H7 x如何在Java应用中将图像存储到图像文件中?  . L( \, L8 [4 x* n( N
0 v9 t! O) ?# G$ ~/ A. c5 i
答:  ) Z) [4 C3 Q1 q
Java Advanced Imaging API(包含在Java Media API中)允许在Java应用中执行复杂的,高性能的图像处理.JAI API提供了存储图像的能力.目前,JAI API支持以下几种图像文件格式:BMP,JEPG,PNG,PNM,TIFF.下面给出了将图像存储到BMP文件的一段代码:  . u, k, M, `) M& t

) [/ f! ~7 Z' d; J* R' [& M8 ?/ E9 d7 L8 a; t: p
OutputStream os = new FileOutputStream(fileToWriteTo);  ) h% Q9 d- b, Q& ?- q
! z0 V  U  O$ n0 R  C2 E
BMPEncodeParam param = new BMPEncodeParam();  
; W6 |+ H! j# T4 E( c5 R8 G. X2 {8 {: c" w& H0 x! G- ]
ImageEncoder enc = ImageCodec.createImageEncoder("BMP", os, param);  8 x+ M& y. |; _
' I" h3 X6 \/ i( h; p4 @) @: n
enc.encode(img);  
: M* {6 P; J1 s! r- u
; e3 b3 D0 H/ sos.close();  
; Q, A& w/ b1 M3 m  m) O+ q. f6 Y: n6 X6 S2 b6 I' d
有关存储图像文件的编程指南请参考以下网页:  
' l8 X6 D4 W. o3 e6 X# D. \& W2 n3 k- q7 B
http://java.sun.com/products/java-media/jai/forDevelopers/jai-guide/  
: j) M/ L9 |$ x! q0 d3 |* s% f4 [" j( Q) d% p

4 s/ g# f$ l: |# K& x/ X6 e8 P/ h/ ?% W+ N) ]+ T) d: R

% n5 s% }& ~( G+ ^问:  ; V+ U9 x6 f8 C5 y! Z" f
如何用Java语言向串口读写数据? font>  2 h- E2 O' N3 D* S2 D0 n6 j0 z

4 ?8 j1 Q( ]+ u5 G$ M; K答:  
( |$ t( P+ S* F1 J" T, eSun公司的Java Communication API2.0可用于读写串口,它支持RS232串口和IEEE 1284 并口,提供了一种与平台无关的串/并口通信机制.  
1 F. V. ^8 A" U
3 Q4 y  o7 Y+ C2 x+ r5 [  i详细文档,请访问:<a href="http://java.sun.com/products/javacomm/" target="_blank" >http://java.sun.com/products/javacomm/</A>
zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
如果我没给你翅膀,你要学会用理想去飞翔!!!
wangyu249        

1

主题

3

听众

54

积分

升级  51.58%

该用户从未签到

新人进步奖

回复

使用道具 举报

wl820609        

1

主题

3

听众

23

积分

升级  18.95%

该用户从未签到

新人进步奖

回复

使用道具 举报

deven1985 实名认证       

23

主题

3

听众

2433

积分

智慧的蓝色小毛驴

  • TA的每日心情
    开心
    2011-12-5 23:03
  • 签到天数: 106 天

    [LV.6]常住居民II

    自我介绍
    200 字节以内
    不支持自定义 Discuz! 代码

    新人进步奖

    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-4-20 05:42 , Processed in 0.467640 second(s), 70 queries .

    回顶部