編程必備經典:Java常見問題集錦

來源:互聯網
上載者:User
編程|問題
問: 如何設定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. 下面給出了這兩種轉換的實現: 

  String native2Unicode(String s) { 
  if (s == null || s.length() == 0) { 

  return null; 

  } 

  byte[] buffer = new byte[s.length()]; 

  for (int i = 0; i s.length(); i++) { if (s.charAt(i)>= 0x100) { 

  c = s.charAt(i); 

  byte []buf = (""+c).getBytes(); 

  buffer[j++] = (char)buf[0]; 

  buffer[j++] = (char)buf[1]; 

  } 

  else { 

  buffer[j++] = s.charAt(i); 

  } 

  } 

  return new String(buffer, 0, j); 

  } 




  除使用以上兩個方法之外,有些JDBC driver如果對jdbc driver Manager設定了正確 的字元集屬性,以上2個方法就不需要了. 

  問:    當用Servlet來處理http請求併產生返回的HTML頁面時,如何使HTML頁面中的中文字元能夠正常顯示? 

  答:    javax.servlet.http.HttpResponse類用於產生返回頁面.通過HttpResponse定義的方法getOutputStream()可以獲得ServletOutputStream的執行個體,這樣使用者就可以利用ServletOutputStream.write方法向輸出資料流中寫入返回頁面的內容. 但是ServletOutputStream使用的是預設的編碼方式,如果要使返回頁面中的中文字 符能夠正常顯示,最好顯示地指定所用的字元編碼方式. 通常需要構造一個 OutputStreamWriter , 常式如下: 

  public void doGet (HttpServletRequest req, HttpServletResponse res) 
  throws ServletException, IOException 

  { 

  res.setContentType("text/html"); 

  ServletOutputStream out = res.getOutputStream(); 

  OutputStreamWriter ow = new OutputStreamWriter(out,"GB2312"); 

  ow.write("這是測試"); 

  ow.flush(); 

  ow.close(); 

  } 


  問:    如何設定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資訊,範例代碼如下: 

  import java.util.properties; 
  ..... 

  Properties sys = System.getProperties(); 

  sys.put("proxySet","true"); 

  sys.put("proxyHost","myHTTP.proxyserver.com"); 

  sys.put("proxyPort","80"); 

  System.setProperties(sys); 

  u = new URL(website); 

  connect = (HttpURLConnection)u.openConnection(); 

  ..... 


  問: 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所在的視窗,作為模式 對話方塊的父視窗. 範例代碼如下: 

  ..... 
  Dialog d = new Dialog( getParentWindow(comp),title); 

  // comp為applet上的任意一個組件 

  .... 

  public void getParentWindow(Component compOnApplet,String title){ 

  Container c = compOnApplet.getParent(); 

  while (c != null) { 

  if (c instanceof Frame) 

  return (Frame) c; 

  c = c.getParent(); 

  } 

  return null; 

  } 




  問: 在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應用 

  問:    在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")的常式如下: 

  try
  { 

  process = Runtime.getRuntime().exec (command); 

  InputStreamReader ir=newInputStreamReader(process.getInputStream()); 

  LineNumberReader input = new LineNumberReader (ir); 

  String line; 

  while ((line = input.readLine ()) != null) 

  System.out.println(line); 

  } 

  catch (java.io.IOException e){ 

  System.err.println ("IOException " + e.getMessage()); 

  } 




  問:    如何產生簽名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檔案的一段代碼: 

  OutputStream os = new FileOutputStream(fileToWriteTo); 
  BMPEncodeParam param = new BMPEncodeParam(); 

  ImageEncoder enc = ImageCodec.createImageEncoder("BMP", os, param); 

  enc.encode(img); 

  os.close(); 


  有關儲存影像檔的編程指南請參考以下網頁: 

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

  問:    如何用Java語言向串口讀寫資料? font> 

  答:    Sun公司的Java Communication API2.0可用於讀寫串口,它支援RS232串口和IEEE 1284 並口,提供了一種與平台無關的串/並口通訊機制.


相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.