在线时间 0 小时 最后登录 2007-11-12 注册时间 2004-12-24 听众数 2 收听数 0 能力 0 分 体力 2467 点 威望 0 点 阅读权限 50 积分 882 相册 0 日志 0 记录 0 帖子 205 主题 206 精华 2 分享 0 好友 0
升级 70.5%
该用户从未签到
问: 如何设置Java 2(JDK1.2)的环境变量? & l$ H( {( z- A2 _) n3 `6 r8 s* S0 l
' [; I# j5 U9 G3 i4 h: N
答: Java 2安装后,需要设置PATH和JAVA_HOME环境变量.与JDK1.1不同的是:设置好JAVA_HOME环境变量后,JVM将自动搜索系统类库以及用户的当前路径.
; K) e# g3 l( O. Q8 n6 a Java 2环境变量的设置如下例所示: 4 u- X4 S- b! O
Solaris平台: setenv JAVA_HOME Java2的安装路径
0 }$ S* w# c( ~ Y% t1 u' C5 o0 | setenv PATH $JAVA_HOME/bin {PATH}
3 m/ ^' B( T7 G$ p' z Windows平台: set JAVA_HOME=Java2的安装路径 8 n3 U- C8 C: h3 r8 ]
set PATH=$JAVA_HOMEbin;%PATH% + f$ _6 q% s% f) L
/ q+ A; V$ |! I: r9 g6 w 问: 哪些Java集成开发工具支持Java 2?
$ P& G6 e, L) g6 j ; K6 B9 C2 L+ D
答: 目前流行的Java集成开发环境,如Inprise的JBuilder,Symantec的Visual Cafe, Sybase的PowerJ,都支持Java 2. 3 h4 n# ]& d# i5 D. B& F
1 c6 j% m% ^! {7 \; ^ _% T* y9 C 问: 如果在Netscape或IE浏览器中运行Java applet时出现了错误,如何确定错误范围? ( d( {& ~$ f% T5 e$ U4 h
. F" ~4 \4 I/ ]; t
答: 当java applet在浏览器中运行时,使用的是浏览器本身的缺省JVM.而不同浏览器对JDK的支持程度也不尽相同. 因此,在Netscape或IE浏览器中运行Java applet出现了错误,建议使用JDK提供的工具appletviewer或Sun公司的Hotjava浏览器来测试该applet,以确定错误的产生是与浏览器相关. * C I: v% I4 a
如果applet在appletviewer或Hotjava中运行一切正常,则错误的产生是由于浏览 器不完全兼容JDK而引起的. 此时,解决方法可以是使用Hotjava浏览器或者安装 Sun公司的Java Plugin.
! e8 l( \$ T, v( X6 ` 如果applet在Hotjava浏览器或appletviewer中运行即发生错误,则应当根据错误 提示检查applet程序. 6 ^1 Q1 A+ E, D6 ^/ s4 H
6 q. y( X0 Y! ~8 |( V
问: 当用JDBC向数据库中插入数据或从数据库中提取数据时,为何有时中文字符会显示为乱码? " l4 L) ~$ N; _
( d- ~0 y0 x% b) {
答: 这个问题的实现通常与各个JDBC driver的实现有关. 目前大多数JDBC driver采用本地编码格式来传输中文字符,例如中文字符"0x4175"会被转成"0x41"和"0x75"进行传输. 因此我们需要对JDBC driver返回的字符以及要发给JDBC driver的字符进行转换. 7 i! z0 L) y: Y1 b3 O
& N9 ?) N& a5 X( L4 d4 _1 G9 b
当用JDBC driver向数据库中插入数据时,需要先将Unicode转成native code; 当 JDBC driver从数据库中查询数据时,则需要将native code转换成Unicode. 下面给出了这两种转换的实现:
3 I2 \( W8 w; v# L' J6 i
4 y5 u7 ^8 q) S# [+ c9 | : z7 b6 q2 A8 X! w7 g' c, J5 h
String native2Unicode(String s) {
" e, p' M4 b, Q% J( ~
1 c4 B2 O5 d6 P6 w' A! c* N if (s == null || s.length() == 0) {
% r' V1 G% t. A/ T1 M( c$ O$ ^ ! ]/ I4 X- i5 ~/ d, Q, z+ Y
return null; ( m% w" S# d- x/ T8 h
' C2 t3 J# T' | _. r9 \0 @9 \4 }
}
3 Q& L. t$ J/ c5 C) a
$ X7 N# ^: A7 v5 q' f2 k; P3 c ? byte[] buffer = new byte[s.length()];
% h. c% ]! f1 Z; s
: Y/ \; G! \4 u6 `% F% m0 g for (int i = 0; i s.length(); i++) { if (s.charAt(i)>= 0x100) { 1 D. Z5 N' m& i
/ m. B+ D1 \8 g, N c = s.charAt(i);
7 E7 V- n1 N3 h" y7 Z# V" Q7 |4 [, j * n. G8 @& q) p2 p! @9 q
byte []buf = (""+c).getBytes();
* ]7 s3 c8 o6 C* y; p7 e; A& i5 o' d8 V ( w- e: W7 j8 w0 F2 s' T1 u5 k7 \
buffer[j++] = (char)buf[0];
1 t2 J7 b( v# x7 l, M
; u) L+ ]( p$ g2 l3 f' t6 W2 T buffer[j++] = (char)buf[1]; q: Q6 j2 O$ f) x
# J9 a) \8 T7 [+ V. k
}
" }( T7 X- I0 a% j' p5 E4 F
% d0 G: }4 r4 M, v else {
! A: a, X& o) w1 v* J . \, m5 H9 ?( `& w
buffer[j++] = s.charAt(i);
1 R5 }9 D) Q0 ~5 b9 @) W/ ]4 k2 v8 ? ) J+ w/ S r( f0 |$ s/ H# }
} / h& U! r' I+ v/ Z, E
% o* W% L% R( e( t- z; e5 T
} 5 c3 s- P+ X. u! @7 G6 u* O
4 m e1 K3 t7 N, Y3 e0 h
return new String(buffer, 0, j); / k O: Q% ^" U' o7 ~$ k% a6 \
, Y0 N! B- n$ D# X3 U6 p8 v0 O1 l
} ) S! R3 b5 T1 `; J
8 q8 n/ h n/ C6 m7 b% P 除使用以上两个方法之外,有些JDBC driver如果对jdbc driver Manager设置了正确 的字符集属性,以上2个方法就不需要了.
# ^, z5 T' }" X! }: u7 ^+ O . u: w6 k, z6 e9 Z7 Y$ |; n, ?
% L3 w4 v# G" h; P3 A7 c3 N
问:
4 c9 v& u3 y; c2 J 当用Servlet来处理http请求并产生返回的HTML页面时,如何使HTML页面中的中文字符能够正常显示?
a6 ^2 Q. ?" ^+ p; n+ e d. |1 P. Z8 ^& p# j: C
答: : j. }2 Z4 K7 Z4 B0 N) J# l
javax.servlet.http.HttpResponse类用于产生返回页面.通过HttpResponse定义的方法getOutputStream()可以获得ServletOutputStream的实例,这样用户就可以利用ServletOutputStream.write方法向输出流中写入返回页面的内容. 但是ServletOutputStream使用的是缺省的编码方式,如果要使返回页面中的中文字 符能够正常显示,最好显示地指定所用的字符编码方式. 通常需要构造一个 OutputStreamWriter , 例程如下:
, x7 E/ }: n& J6 G o. h8 c
4 _( Z) [3 U9 s; T* v; Z2 e9 Y 0 L, `7 B9 e- M4 M
public void doGet (HttpServletRequest req, HttpServletResponse res) 8 p* X% ~9 p5 _
* e: U' q/ N* N# a7 d" l2 y& C/ T$ r
throws ServletException, IOException 1 o1 g3 C" @4 N8 l" g5 n+ V" l
7 j+ Y' w* m! S+ ]* `5 |& ] { 4 i" b2 ]; g2 i
: K2 r/ X/ w% X* A C res.setContentType("text/html");
7 Y. W* H3 R. p' e. j
7 Y2 k( I( e" M" ^) R: O ServletOutputStream out = res.getOutputStream();
& w8 h/ q, D+ J' e9 P8 k+ y 2 u0 s& \7 ?! Q" X
OutputStreamWriter ow = new OutputStreamWriter(out,"GB2312"); ! n8 w _) E- a+ r
0 I |) e" ]( V" w8 A+ o! L3 I
ow.write("这是测试");
5 }( @- `) P' r7 C0 E y! O+ B; |3 e
/ A. c6 {" l: g$ @7 T ow.flush();
* d* \, e7 [7 E" L( K0 w% O
0 I0 c+ ^0 P1 F0 w# r/ I ow.close();
9 e2 d9 s3 _) V0 d5 b- t- ^
3 k; O- X7 N* Q* I }
4 ^9 o6 [$ S+ a1 N# \) M# ~
" N+ j6 E0 b, j" I) U' ^0 T; q3 E
# g x9 a0 ~$ s4 f 问:
8 R$ B, N6 T; P; M 如何设置Java WebServer的CLASSPATH,以包含用户的class文件? ) E) J: U# i5 \. i0 \
! T+ }9 c) M0 J$ J 答:
1 {* I; G" h3 @. ^7 x 有两种方法可以设置Java WebServer的CLASSPATH环境变量,以使用户编写的Servlet能够调用用户的class文件.
l4 g. Y5 ^/ I( y+ L) _& K3 O 将用户的class文件放到 JavaWebServer_Dir/classes目录下,在Java WebServer 启动时,classes目录被自动加入到CLASSPATH环境变量中了. 0 h: M3 Z$ w5 f$ W, Z* [* |
修改httpd.nojre文件,将用户class文件所在的路径名加到CLASSPATH环境变量中.
# l. r0 p' a" t2 Y/ ? + R- g8 q% E4 o( n" p+ F
/ G4 v1 S; w3 L+ F) L) w8 O% F
问: 8 [7 y7 F+ t2 V4 N! P
为什么在Windows平台上用Naming.lookup来获取远程RMI对象时会很慢?
$ z: @0 i# Y- d0 i- w! @. c: x) G
0 o3 [& R* U* m7 { 答: " t- b% s* B# P- B; d4 a
机器的网络设置不正确很可能会引起该问题的发生.
/ F1 L- K$ L! Q% |! L. R RMI使用了Java网络类,特别是java.net.InetAddress类,它将查询TCP/IP的主机名, 包括IP地址到主机名的映射和主机名到IP地址的映射.在Windows平台,这种查询功能 是由本地的Windows Socket库来实现的. 因此延时是发生在Windows库中,而非RMI中. 9 S8 T& P/ [7 u7 A9 U" S
( g/ Q/ W6 N1 z5 F7 } m7 ] 如果你的机器设置成使用DNS,问题通常是DNS服务器查不到主机名,你所发现的延时 是DNS查询的延时. 请尝试将RMI通信中涉及到的所有主机名/IP地址加到本地文件 winntsystem32driversetchosts或windowshosts中. 格式如下: " B$ @* T$ |' [) p* ?
2 e. D( f1 R+ t5 F; c) L# t7 Y
IP地址 主机名
3 q' s, m6 R+ G& i& `( _
7 T- }* E5 Y& ~5 n- d( Q2 Y 如此设置应当可以明显地减少查询所花的时间. ; y& x9 }9 }1 {" X" Y
3 v1 _! Y0 U" i% f9 E7 {
问: 编写Java application时,如何设置proxy的信息,以便访问外部网站? & T# G* z3 u9 H; a- x1 \
1 L% B }0 e* T
答:
& h' D4 f( t a: g- G0 s1 A0 y 若在java application中访问外部网站,首先应设置proxy信息,样例代码如下: 5 v& K- a4 _' E- \
4 L! G3 Z) l1 e: y) ^1 U
( o+ a: Y5 Z0 U- F import java.util.properties; F0 }2 K6 f7 G K
, L$ l. E0 N7 q* P0 l ..... 4 @1 K, E) p7 y8 x+ z* Y, m; S
6 C* \$ ^4 C/ O1 Y) b$ m
Properties sys = System.getProperties();
& Z/ P$ t% I. e
) f" [# _1 C: U+ l+ g {9 C$ ~ sys.put("proxySet","true");
& N, [/ R) V! r7 }7 X
$ s4 h' T- @2 ` sys.put("proxyHost","myHTTP.proxyserver.com"); 8 I* S, o" ~5 w! k( l
9 C4 o6 r7 D h& D sys.put("proxyPort","80");
0 f5 E' d$ `7 m3 Y- o9 F
+ U+ h( s! K# p7 N$ X System.setProperties(sys);
& q# V, y4 k( G) R) J3 T
$ L* f* s2 O( [: K1 u( w6 [
4 Q( u3 ~& B! c2 W# ~" m
# c: T+ E+ T: u$ c) b, h u = new URL(website); : r7 P& {7 t# U0 t/ C1 I( y
8 l7 f0 T" k! [9 H3 d: _, l6 `
connect = (HttpURLConnection)u.openConnection();
) w1 N# W9 I* R
: z1 x7 n+ D# _9 r( E ..... ! ?# P$ \, V& W; I2 Q/ r
# r. h4 { u7 ?! a( A* T9 A
问: Swing组件JList的列表数据修改了,如何通知JList改变显示? 4 h5 D* \1 u0 [8 i# p, A
8 s Z+ f- P1 Z
答: ' L" S9 t! l' K. G/ h/ {
JList组件有一个单独的显示模式ListModel来表示JList的显示数据.
j/ {) c- Z4 f+ c+ M g0 n; r2 ? JList创建以后,JList数据元素的值及数据元素的数量可以动态地改变.
6 Y% W, T- W2 U& U JList在它的数据模式ListModel中观察数据的改变.因此,一个ListModel 的正确实现应当在每次数据发生改变时,通知事件的监听者.
! q3 r2 W/ s0 A, B& h 当使用构造函数JList(Object[])创建一个JList的实例时,系统将自动 创建一个DefaultListModel的实例来存储JList的显示数据, 可以调用 DefaultListModel中定义的简便方法来动态地修改JList的数据,如 removeElementAt(index),addElement(Object)等. DefaultListModel 在修改数据的同时,将通知JList关于数据的改变.
5 ~" i1 `$ z: }
, a( e) M4 d) Y' H' h6 f 问: 5 A9 O2 [! [+ p3 R
在Java applet中如何实现一个模式对话框? # h8 W( n& i4 t& D( Y5 {& N: u
" o( T) T$ L( W1 X8 w 答:
/ M& ^/ k; @- r0 ^+ ? ` 在Java applet中实现模式对话框的关键就是在创建一个对话框的时候 要为该对话框指定一个正确的父窗口.因为Applet是Panel类的子类,不 可以作为对话框的父窗口,所以首先要获得applet所在的窗口,作为模式 对话框的父窗口. 样例代码如下:
7 {6 r1 M0 X8 k$ D) f 2 Q# Q$ `- ^ E' z
* ~. b6 h! R7 Y8 V/ [. V: i, P
..... 1 A- f! i9 r5 g3 Z- g* E: e7 l
4 ?! X% L+ j; y: s% F9 {* c Dialog d = new Dialog( getParentWindow(comp),title);
7 B8 P" D7 E3 |! r' D 4 i# \$ E& n1 K6 H
// comp为applet上的任意一个组件 1 g' b: p$ K% e3 X
8 Y" S6 K* `: J: x
....
; | t/ D* |" ~' n& L , L) `7 Y5 k* ^! V. q
0 n* `6 y2 \* ^$ |8 b- L
) k, b; h8 z$ |
public void getParentWindow(Component compOnApplet,String title){
7 m! Q. I: i7 w2 N3 S
; L( D9 e- m; W3 h2 h Container c = compOnApplet.getParent();
2 V$ ~- m8 C+ k$ f; N 3 Z) n( \. o( F2 a/ h6 |
while (c != null) {
+ D# J9 [! s7 m! U5 ^# ?
9 R: P7 D" p8 ?9 f1 X$ T# ]1 @5 |* L if (c instanceof Frame) 2 D R& p' s2 J8 w4 J
+ r' c) @! I; \
return (Frame) c; , n, _0 n( g5 w9 ?$ u1 [
, f! u" U2 o2 [; |
c = c.getParent(); , ]4 q; {! W8 T& e" G4 _
( V z$ Z: G' x8 C8 p% j+ C S
} ) ~: E8 V; k3 U D( E
8 ~! l w" ^* n8 ]2 P/ d" _ return null;
! [: a) f7 ~9 ~- A2 T
0 l4 ? G* V3 J: I+ X. O% A }
- Z3 t, l2 _- e3 e# R4 a/ E
# p$ I6 G m: L- D 问: 在Java applet中如何显示另外一个HTML页面? & N- u2 q2 I: |5 `
" h' T) k7 Q6 {6 @! N4 ^0 R 答: 0 ~4 c8 d U2 T" }
通过java.applet.Applet.getAppletContext()方法可以获得与该applet相关的AppletContext, AppletContext.showDocument(URL)方法就可以使applet所在的浏览器显示另外一个网页.
" r/ T6 ^" D1 A; `2 O& p1 Q2 e
# h" v3 J6 n: Y" Y1 u 问:
7 O) B( F$ r: e0 M2 H/ ^ 用JDK实现的签名applet,可否在Netscape或IE中运行? / t$ y: g! {, _ j5 s
, j8 X2 M$ w, @' T/ _' _( _1 i! B
答:
8 m a: J" F8 I3 ] 用JDK实现的签名applet,不可以在Netscape或IE中运行,但是可以在Hotjava浏览器中运行. `# T+ O2 L+ H F$ C3 U
8 Y' y$ d5 a2 M2 o& D( J! R
不同的浏览器提供了不同的签名applet机制,如Netscape提供了zigbert工具和 Capability API, 而IE则需要使用CAB文件. 但是,无论是Netscape工具产生的 签名applet,还是用IE产生的签名applet,都不可以在其它的浏览器中运行.
* }6 O0 a8 ^. w " E4 r1 p/ K. l- c
如果要使JDK产生的签名applet能够在Netscape或IE中运行,解决方法是在 Netscape或IE中安装Java Plugin,则用JDK实现的签名applet就可以在这两种 浏览器中运行. ' \0 c6 z" `0 q" ~- I& ^2 G
: M3 p9 k; |+ l! S2 N( q8 S5 b
问: . h2 p7 [# z% s( \- L. [/ |' |9 R% p
用JNI技术可以从Java应用中调用C程序库,但是如何使该C程序库可以调用另外的C程序库?
7 p' J$ ~# U) O2 h x" _# [
9 k/ x- p+ k/ c `3 O" s 答: 4 }( }$ F! H) h$ B% [
如果一个被Java调用的C程序库C1仍需要调用另外一个C程序库C2,那么在编译C1的时候应当联接程序库C2,步骤如下(Solaris平台):
) e- c# ]/ [4 f 编写调用C库的Java文件,并编译. 4 f- @" c2 I- @) Z
javac java文件名
$ A6 V' S/ T- b y( t ; P- ~: ^! ]2 R( N
% |* _4 R, D _/ _7 e* C
产生C程序头文件 / m$ K9 o- G6 Z, n: |
javah -jni java文件名(不带后缀.java) 6 G% g- ^9 |5 Q
6 D, a) B0 G* z0 x # u! N5 o: q, Z1 o
编写被Java调用的C程序C1.c,以及被C1调用的C2.c,并编译.
# c; k. t; h+ a cc -G -Iinclude路径名 C2.c -o libC2.so 1 D/ D$ O& b6 N
cc -G -Iinclude路径名 -lC2 C1.c -o libC1.so 6 {" H& v* C" p' y' N: z. ~/ w
0 x1 x& [, t1 ]7 I. n& {
0 V5 D/ g. A# o% |. V! r 设置环境变量
6 O: m- B. n3 D) Q D setenv LD_LIBRARY_PATH libC1.so,libC2.so所在路径
% \% {$ d, c4 B6 t: {/ P {LD_LIBRARY_PATH} # ]5 V4 ?7 _: @
, j. p5 @, O5 l 9 X* |* `/ z; x, F8 }. ^& [9 ]1 H; e
运行java应用
8 n% y; p( i# O9 ?0 l - y* K! u1 I& q4 V% Q1 q5 G
- f8 J; N! B v( _" S/ e! n 问:
" G, K: b0 F9 C [+ l3 h, Z( F/ b 在Java语言中,如何列出PC机文件系统中的所有驱动器名?
' k2 Z; s/ J. o4 y% b' Y
9 C6 t' [. b; x! H* K0 t 答:
@ f3 b, l+ X# } 在Java 2版本中,java.io包中的File类新增加了方法listRoots()可以实现这一功能.
$ ~+ c- d6 `9 {9 e) Y8 o" C0 f
+ ?! D0 ]4 Q1 ]) a6 S 问:
7 f. e9 \( u$ X& I e9 \ 为什么Runtime.exec("ls")没有任何输出?
( }# r0 G( U& F# K+ g$ _1 j 0 f, g) ~8 G% U) F0 j$ I
答: 3 {; b/ d7 I l: |$ z
调用Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息. 由于调用Runtime.exec方法所创建的子进程没有自己的终端或控制台,因此该子进程的标准IO(如stdin,stdou,stderr)都通过Process.getOutputStream(),Process.getInputStream(), Process.getErrorStream()方法重定向给它的父进程了.用户需要用这些stream来向 子进程输入数据或获取子进程的输出. 所以正确执行Runtime.exec("ls")的例程如下:
" r6 N( |- g1 a1 G$ F( j; p & R5 f! W @5 h% ^: |+ j/ J
& C C2 m: x0 v: Q2 A try ) J, v& m; c: B
* t' k& ]% d" U: ~ {
) A6 e; } [! K L$ L& x0 J1 I
0 T4 s4 [4 {4 `. A. H$ j$ T& ~ process = Runtime.getRuntime().exec (command);
1 c# k) }( ]7 T( r6 q: w- g
' O9 i& H4 m1 a0 S8 W InputStreamReader ir=newInputStreamReader(process.getInputStream());
6 n K/ C+ a3 Q( d/ X& W9 r ; h. Q" _1 l, i6 p
LineNumberReader input = new LineNumberReader (ir); 0 V' E3 D- L, s& a% {0 Y. m! @: Z
. p2 G2 ~7 C* t! P5 W4 _
String line; 8 c9 {1 |5 M7 p" O
6 N. K* A. v) W6 Q9 ^5 j while ((line = input.readLine ()) != null) 1 f- S1 R' `# X5 _4 }( n. [
# u! S; v- p5 V* P0 m* N System.out.println(line); : G8 ], ~/ m1 _+ i7 Y
/ |" v. E# I# Y# m ~8 R) y/ P
}
D9 Y/ L) U$ e+ S+ ?: @ 0 s8 d8 t: @* C2 K4 p3 z+ }2 }7 ?3 x1 t
catch (java.io.IOException e){ / W$ D7 M. t6 I2 x
7 x/ f e% `8 Q) n8 @
System.err.println ("IOException " + e.getMessage()); ; j. a b* ~* q5 I5 b. z/ L
& ]6 z/ o. n6 I* O" o0 q }
) k: ~4 J& s! x+ M
7 m8 {- L2 y. o3 g2 ^8 I 9 w9 d4 G. `) f* ^
问: ( j+ a. v/ e) [$ k) `
如何产生签名applet,以使applet能够访问本地资源? * w8 P) L! R! l
( |0 E% e1 r% I
答:
8 C1 l" z7 z( b9 N7 j 在jdk1.1中,可以使用javakey命令来产生公钥,私钥,证书和签名的jar文件,详细资料 请参考: http://java.sun.com/security/usingJavakey.html而java 2对签名机制做了比较大的改进,允许用户更灵活地设置安全权限.Java 2提供了三个工具:keytool,policytool和jarsigner来实现签名applet.例如,Joe编写了一个签名applet:SignedApplet.java,那么产生一个简单的签名applet的过程如下:
9 f2 ~& q1 |8 w4 C
( u) i% o: _3 A* ^6 r
) x& _2 w1 F) ~0 I3 v2 v //产生密钥,密钥别名为joe,口令为sign12,存放在密钥库joestore中
, p* y, a" c% z5 a$ U
4 J, x1 l$ F: v1 Y! I1 E4 G keytool -genkey -alias joe -keypass sign12 -keystore joestore
" A, u. M/ V e, `
1 [( a, j' ]! i6 J% d- E //将SignedApplet.class及相关文件打包成jar文件 , I- R2 x6 U0 E, b% _( K
5 r' O: F' O+ I1 h1 J jar cvf SignedAppletDemo.jar 3 X' S' s4 P, g5 b8 B
0 r0 X9 y; K( V7 H
//利用keytool生成的自签名的证书产生签名applet(jar文件)
8 o# u2 i0 S$ Y$ \9 y " q0 z5 z& v8 @% j2 Z; M
jarsigner -keystore joestore -signedjar joe.jar SignedAppletDemo.jar joe
- X5 B; f3 j/ z: Q q
3 V: V) o* e* B- c R4 q8 F# @ //将自签名证书从keystore中输出到文件
9 S# ?' n8 v. z' o7 S, c% k 2 m( m& h& N, _% y" V$ O+ Q) ^8 m
keytool -export -keystore joestore -alias joe -file joe.cer
9 J" Q% M+ W2 S, r" m# D
' H% @! |) U7 z- l: k) n
/ Q% I' A9 P7 @" D- b H. E" q+ e
/ A+ [, o* s. r; j1 C8 n 而对于签名applet的接受方Susan,需要通过如下步骤来安全地执行
* _: }3 m! t3 F. D " l2 U) _2 n8 i% _
Joe编写的签名applet:
0 A9 o1 r+ p9 ]4 ]1 e / m) W# S5 B6 P8 Z
//得到Joe的证书并将之读入到密钥库中susanstore中
5 {$ m' ~' | H
4 Z: N7 _' i6 H( B) U8 O keytool -import -alias joe -file joe.cer -keystore susanstore 4 z t1 V5 k6 R
' U% d: b0 j! V. G8 h //运行policytool产生满足Susan要求的policy文件
0 |; l& x8 r. m
" Y6 x' T3 q2 ]# d! Z2 f policytool
6 K' w( {" i; f( x* H 9 A s- d5 b! R; K$ t: p- x4 Y. N) k
//用appletviewer运行之,或在浏览器中安装java plugin来运行之. 5 x2 r& a, K( B( U {4 S
r" R. @' t9 F+ I* [! { 3 w& H) l9 c$ \8 J+ m' F, ]0 N0 m
8 N+ t; m( V. D4 S8 P/ W2 s# t
关于签名applet在Java Plugin中的部署请参考以下网页:
3 s9 |% r- [! e a, c# _# g: E0 W
: j/ M! Z9 n1 I http://java.sun.com/security/signExample12/
5 {1 v6 L1 d8 i2 }* f0 | ! U& z+ }: V1 y: o2 |
) j# n8 t- A9 Z3 Q' F
0 v3 q. v7 D& f `. b
注:以上的例子为简单起见,使用了keytool产生的自签名证书.其实,用户也可以 2 P/ }6 L% v4 V8 t) I( o
' K2 [3 u- m8 z. a7 S$ v$ m
使用keytool -certreq向商业CA中心申请电子证书.
/ p% y& W! G$ s ; n- }5 X8 D7 i7 M: c' t/ |1 M Z
6 ]: G6 \* Q# Y0 {! B$ G1 o + E6 d8 V% ?0 W" `8 L+ i( U
2 w. C/ {/ F% U! m 问:
* k. s3 d3 _1 a9 I: z% F 若通过ObjectOutputStream向一个文件中多次以追加方式写入object,为什么用ObjectInputStream读取这些object时会产生StreamCorruptedException?
& F m9 _5 P+ q' j! N $ g9 y5 I | q1 f% v I
答: ( I% o) i0 ~& o D" M/ H2 g o
使用缺省的serializetion的实现时,一个ObjectOutputStream的构造和一个ObjectInputStream的构造必须一一对应.ObjectOutputStream的构造函数会向输出流中写入一个标识头,而ObjectInputStream会首先读入这个标识头.因此,多次以追加方式向一个文件中写入object时,该文件将会包含多个标识头.所以用ObjectInputStream来deserialize这个ObjectOutputStream时,将产生StreamCorruptedException.一种解决方法是可以构造一个ObjectOutputStream的子类,并覆盖writeStreamHeader()方法.被覆盖后的writeStreamHeader()方法应判断是否为首次向文件中写入object,羰?则调用super.writeStreamHeader();若否,即以追加方式写入object时,则应调用ObjectOutputStream.reset()方法.
8 Q* _% @$ \9 |" }) A% r G: }
# _* D: K% k% N' \1 j# Z3 y2 ?8 h 问:
, c- u! j6 D5 s1 p 对象的序列化(serialization)类是面向流的,应如何将对象写入到随机存取文件中?
% A% S5 j% z; e/ c% @. A
) k+ d ?) z3 z, k& J% | 答: $ N4 H$ Q: q) [
目前,没有直接的方法可以将对象写入到随机存取文件中. 8 u$ d+ w' o" G
但是可以使用ByteArray输入/输出流作为中介,来向随机存取文件中写入或从随机存取文件中读出字节,并且可以利用字节流来创建对象输入/输出流,以用于读写对象.需要注意的是在字节流中要包含一个完整的对象,否则读写对象时将发生错误. 例如,java.io.ByteArrayOutputStream可用于获取ObjectOutputStream的字节流,从中可得到byte数组并可将之写入到随机存取文件中.相反,我们可以从随机存取文件中读出字节数组,利用它可构造ByteArrayInputStream,进而构造出ObjectInputStream,以读取对象.
5 ?2 X3 E# s% ~" }
0 Y4 I( E& E [1 c( H& q- `9 w 问:
7 g O* a+ f8 C( u0 t+ r- n 运行RMI应用时,可不可以不手工启动名字服务rmiregistry,而是从程序中启动之?
: o" s- p! V. [7 a# I+ [5 N: h% v2 w
) R! y6 C5 D" B# U, o( D 答:
5 |- _/ e, ?5 w' G" t: B 可以. java.rmi包中提供了类java.rmi.registry.LocateRegistry,用于获取名字服务或创建名字服务.调用LocateRegistry.createRegistry(int port)方法可以在某一特定端口创建名字服务,从而用户无需再手工启动rmiregistry.此外,LocateRegistry.getRegistry(String host,int port)方法可用于获取名字服务. % {0 G! c, I3 r4 ]3 t* C) o
6 ~7 `' y$ g+ p/ f+ U }( V5 q 问: 3 c+ a% D# ^$ Y$ Z
使用类PrintJob进行打印操作时,应如何设置打印机名等打印属性? 2 w, O* F0 j. a, l; ]
: v, I8 _: `. J
答: 0 t9 @; ], }; O8 V" K+ } V
使用如下方法可以获得PrintJob的实例用于控制打印操作:
: [1 L7 A7 l g7 t; v- _$ C1 e
$ S2 n- C$ t+ p/ h1 Z
5 g) w8 _2 y Z& E. N9 H' a Toolkit.getPrintJob(Frame f, String jobtitle, Properties prop) 7 U5 A) P: {: d5 W; F0 c9 x0 _
+ I7 H' s. k% N! |* [
那么对于打印属性的设置可以通过对prop的属性设置来实现,打印属性包括:
8 E8 D6 R3 d2 i1 |' q
$ ]5 S9 b" `' w awt.print.destination: 可以是"printer"或"file" 7 V( n* |9 l. J( A0 q0 T. p
6 L+ `8 T5 f' h% X9 v* `* o. T awt.print.printer: 打印机名 # N% L9 R( a: M0 y; ^- |1 N
2 h# t" m6 [5 H9 P! P( y) F8 s awt.print.fileName: 打印文件名
) V% e1 {+ V @# Q, g 2 J# T9 N% }' o( v8 V- G
awt.print.numCopies: 打印份数
! W( z/ R6 i% r& R. |$ M 9 P. v4 t& k$ r" c+ ^
awt.print.options: 打印命令的打印选项
; f/ F) ?. K2 [' Z- O/ p 9 A E) e8 l& U! V7 P
awt.print.orientation: 打印方向,可以是"portrait"或"landscape"
/ y+ }" m7 Q7 s2 p3 ?+ D# k / C" w, l+ J+ F# g
awt.print.paperSize: 纸张大小,可以是"letter","legal","executive"或"a4"
* e, W, Z: `# ^8 U( I! ]: _2 } ' [' M, J3 \) K( {! H2 {
; l! P+ N2 M9 m; O# ^4 ?
L& C) c1 ?; ]3 ~8 ~
4 R9 W3 z+ n0 L! Z/ a; m3 [* {
0 k6 e2 \" ^$ m ; t4 P# O' K% {5 X) o5 M" K) g" u+ ]
问:
2 N& O m0 _6 i( w# o+ f. Z 在JDK1.1中Thread类定义了suspend()和resume()方法,但是在JDK1.2中已经过时,应使用什么方法来替代之? # s$ n2 |! H/ h
0 u5 ]# x( e: u0 o1 s
答:
! M, b5 u6 T3 U/ W Thread.suspend本身易于产生死锁.如果一个目标线程对某一关键系统资源进行了加锁操作,然后该线程被suspend,那么除非该线程被resume,否则其它线程都将无法访问该系统资源.如果另外一个线程将调用resume,使该线程继续运行,而在此之前,它也需要访问这一系统资源,则将产生死锁. / F4 ^, n% m/ n8 J
" g$ O: i3 ?+ X) K9 h- @3 f
因此,在Java 2中,比较流行的方式是定义线程的状态变量,并使目标线程轮询该状态变量,当状态为悬挂状态时,可以使用wait()方法使之处于等待状态.一旦需要该线程继续运行,其它线程会调用notify()方法来通知它. - _5 C3 J& y# x* l) H0 X: u H# T
: ^0 q+ t% j' ^& Q 问:
+ M Q! Y/ T' ]: m, K" \) Q' T 使用JDBC编程,应如何控制结果集ResultSet的指针,使之能够上下移动,以及移动到结果集的第一行和最后一行? ' k# C; B6 q' b1 e: r" a
- ]' }9 X% S w7 q 答:
( `" m/ U8 u9 ~ 在JDK1.1中,ResultSet类中只定义了next()方法支持数据指针的下移.但在Java 2中,ResultSet类增加了如下方法支持数据指针的移动,包括: 6 y3 ?3 N2 ~& ?# f7 y& s$ I8 c
& N4 z+ J( Y* d8 a0 w& v/ z
# r7 U' j2 u9 ]$ C% ? ResultSet.first():将数据指针移到结果集的第一行 ( g, h7 c4 Y: w) Z% J9 @; M5 H# c
* o) E6 k- y3 d4 N" i! f ResultSet.last(): 将数据指针移到结果集的最后一行 ' z, m( B/ \: R# ^4 O; y u
; X0 B+ R) {! U& D; W8 X8 }
ResultSet.previous(): 将数据指针上移一行 $ T) a, N1 G, M
9 F K5 y: d/ I- N
7 l4 i/ c6 f* y/ w* x6 x; n. Z0 ^ 以上的方法定义在JDBC2.0的规范中,所有支持JDBC 2.0的JDBC驱动程序都可以支持上述方法.目前Intersolv和OpenLink等JDBC驱动程序厂商均有产品支持JDBC 2.0 . $ ?4 O% u! K6 w2 i' \
' h5 s9 w8 K) H! @4 H/ M/ f
6 B9 R3 x: G3 Q/ V
问:
" G! K) b, b S2 q 哪几种Web Server支持Servlet?如何使IIS支持Servlet?
7 c* \/ [" u" X! ?6 g, f + A) h q0 D, ?- e
答:
w9 k- Q, @ I) { 目前,支持Servlet的服务器端产品主要有: Sun公司的Java WebServer,Lotus DominoGo WebServer,BEA weblogic Tengah Server,Jigsaw,NetForge,AcmeServer和Mot Bays Jetty等.
; \, G, P+ b3 o% F
6 b" h1 [2 U7 p+ K( t4 V- U 此外,一些第三方厂商也开发了Servlet engine,以使其它WebServer(如Netscape Web Server,IIS等)能够运行Servlet,如LiveSoftware的Jrun(http://www.livesoftware.com/ products/jrun/)等.
5 }, _1 H% S) L3 ]; U; ^
5 [6 e# N/ Q6 d 问: " F8 |8 \1 f6 q7 S, g7 t
如何在Java应用中将图像存储到图像文件中? 4 K2 K) H. E) K
G: p# L& H% W9 r+ E 答: ) L4 M9 ?, W* f( [2 K8 i e2 M
Java Advanced Imaging API(包含在Java Media API中)允许在Java应用中执行复杂的,高性能的图像处理.JAI API提供了存储图像的能力.目前,JAI API支持以下几种图像文件格式:BMP,JEPG,PNG,PNM,TIFF.下面给出了将图像存储到BMP文件的一段代码: ) c- a& l# c3 H; ^8 @' E% n
1 o9 I) k$ |$ N" ?' b+ P
) _. B q3 y: _" } a* H OutputStream os = new FileOutputStream(fileToWriteTo);
: {3 y7 _1 W0 [) q3 E5 ?
- U u! e( L/ ^+ j: b3 l% D8 T' m BMPEncodeParam param = new BMPEncodeParam(); * v3 F8 G9 {& r# h
9 X: ^% S2 b9 j/ r$ ~ ImageEncoder enc = ImageCodec.createImageEncoder("BMP", os, param);
% l1 o" S @6 v) ~6 \* M - O4 V0 ^ z: a; L
enc.encode(img);
: C! ~: ~& a3 i1 b4 J ; Q6 P" b9 J& F; \: l4 F
os.close();
$ I1 w& m; c: C1 R$ \: M7 O . }1 E5 d$ K3 G3 J7 |: V _& k
有关存储图像文件的编程指南请参考以下网页: ) p) c5 a2 p. D B: f7 r0 z
0 L& f" l' I9 l9 ?3 {
http://java.sun.com/products/java-media/jai/forDevelopers/jai-guide/ % Q% Z& F% i. |( `
2 W2 [7 `8 }9 G9 v ; R, Q2 K& W+ s% c+ \& Z4 @
1 j+ {3 i z z' U
# ?: ?4 B$ Z) }8 D; h h' K 问: - F8 T3 ?( @3 _" `4 }: x
如何用Java语言向串口读写数据? font>
9 x0 e8 [& a# ~# A6 x: {
$ k! j/ j6 C& \9 Y$ J8 c1 D 答:
& i0 w( @) y8 t; ? Sun公司的Java Communication API2.0可用于读写串口,它支持RS232串口和IEEE 1284 并口,提供了一种与平台无关的串/并口通信机制. . i6 a1 X6 D% r# P, c9 G! b
& K6 ?5 T" d/ [. N: _ 详细文档,请访问:<a href="http://java.sun.com/products/javacomm/" target="_blank" >http://java.sun.com/products/javacomm/</A>
zan