隨着越來越多手提電話和個人數字助理開始融入到信息高速公路之上,從移動設備上訪問Web站點變得越來越重要。只有你實現移動設備與非移動設備互相通信的功能後,你設計的移動設備應用程序纔可以稱的上是真正有用的。在本文中, 我們將通過一個簡單的例子來學習如何實現手機與Servlet的網絡通信。該程序經tomcat4.0.6、J2ME的MIDP簡表及SUN的J2ME的無線應用程序開發工具包編譯、配置和測試通過,同時在手機模擬器上中文顯示正常。
一些參考書都講了J2ME的網絡編程,但我感覺大都有點雷同,並且很少有完整的手機客戶端程序以及Servlet服務器端程序,雖然手機與Servlet通信的方法有多種,但經實踐與比較,我覺得手機和Servlet通信比較簡單的方法就是DataInputStream.readUTF和DataOutputStream.writeUTF的配對使用,比如手機發送數據可以在Connection的輸出流中用DataOutputStream.writeUTF依次寫入多個參數,相對應的,Servlet打開請求的輸入流(request.getInputStream)用DataInputStream.readUTF來依次讀出。返回的參數也一樣,Servlet用response.getOutputStream打開輸出流,並把返回值依次寫入,手機端打開Connection的輸出流並讀出即可。至於多個參數多個返回值都是沒有問題的,只是多次使用writeUTF和readUTF罷了。讀者可以參考程序中的註釋,以及運行結果來理解手機與Servlet是如何傳遞與返回參數的以及是如何來進行網絡通信的。
手機客戶端:
import javax.microedition.midlet.*; import javax.microedition.lcdui.*; import javax.microedition.io.*; import java.io.*; public class ClientApp extends MIDlet implements CommandListener { Display display; TextField tf1,tf2; String tf1Str,tf2Str; Form inputForm,returnForm; Command cmdSend,cmdBack; final static String defaultURL = "http://localhost:8080/examples/servlet/ServletApp"; public ClientApp() { display = Display.getDisplay(this); tf1 = new TextField("input first param:","盧東方",20,TextField.ANY); tf2 = new TextField("input second param:","王桃羣",20,TextField.ANY); cmdSend = new Command("Send",Command.SCREEN,1); cmdBack = new Command("Back",Command.SCREEN,1); inputForm = new Form("pls input the param:"); inputForm.append(tf1); inputForm.append(tf2); inputForm.addCommand(cmdSend); inputForm.setCommandListener(this); } public void startApp() throws MIDletStateChangeException { display.setCurrent(inputForm); } public void invokeServlet(String url) throws IOException { HttpConnection hc = null; DataOutputStream dos = null; DataInputStream dis = null; try{ hc = (HttpConnection)Connector.open(url,Connector.READ_WRITE); //設置請求屬性 hc.setRequestMethod(HttpConnection.POST); //設置爲POST請求方式,默認的請求方式是GET hc.setRequestProperty("IF-Modified-Since","15 Oct 2003 08:47:14 GMT"); hc.setRequestProperty("User-Agent","Profile/MIDP-1.0 Configuration/CLDC-1.0"); hc.setRequestProperty("Content-Language","en-CA"); hc.setRequestProperty("Content-Type","application/x-www-form-urlencoded"); hc.setRequestProperty("Connection","Keep-Alive"); //Connection頭可以控制MIDlet和Web服務器之間保持"keep alive"特色。 "keep alive"特色是指在MIDlet和Web服務器間始終使用同一個HTTP連接來多次傳遞數據 (在通常情況下,HTTP是無連接的協議,每次數據傳輸完畢後都將斷開連接, 而下次傳遞數據之前將重新建立連接) //發送請求參數到servlet dos = hc.openDataOutputStream(); dos.writeUTF(tf1Str); dos.writeUTF(tf2Str); //用於發送請求參數給servlet System.out.println("手機傳遞給servlet的第一個參數爲:"+ tf1Str); //主要起調試的作用,調試的結果將顯示在WTK的控制檯中 System.out.println("手機傳遞給servlet的第一個參數爲:"+ tf2Str); dos.flush(); dos.close(); //接收servlet響應數據 ds = new DataInputStream(hc.openInputStream()); String return1Str = dis.readUTF(); String return2Str = dis.readUTF(); System.out.println("手機接收到servlet端傳來的第一個參數爲:" + return1Str); //主要起調試的作用,調試的結果將顯示在WTK的控制檯中 System.out.println("手機接收到servlet端傳來的第二個參數爲:" + return2Str); returnForm = new Form("返回的結果"); returnForm.append(return1Str); returnForm.append("/n"); //將返回的結果append到resultForm中 returnForm.append(return2Str); returnForm.addCommand(cmdBack); returnForm.setCommandListener(this); }finally{ if (dis != null) {dis.close();} if (dos != null) {dos.close();} if (hc != null) {hc.close();} } display.setCurrent(returnForm); } public void pauseApp(){} public void destroyApp(boolean unconditional){} public void commandAction(Command c,Displayable d){ if(c == cmdBack){display.setCurrent(inputForm);} if(c == cmdSend){tf1Str = tf1.getString(); tf2Str = tf2.getString(); try{ invokeServlet(defaultURL); }catch(Exception e){System.out.println(e.getMessage());} } } } |
Servlet服務器:
import java.io.*; import java.sql.*; import javax.servlet.*; import javax.servlet.http.*; public class ServletApp extends HttpServlet{ public void doGet(HttpServletRequest request,HttpServletResponse response) throws IOException,ServletException { response.setContentType("text/html; charset=GBK"); //設置響應屬性 //接收客戶端的請求 InputStream is = request.getInputStream(); DataInputStream dis = new DataInputStream(is); String tf1Str = dis.readUTF();String tf2Str = dis.readUTF(); System.out.println("servlet端接收到手機傳來的第一個參數爲:" + tf1Str); // 主要起調試的作用,調試的結果顯示在tomcat的啓動DOS窗口中 System.out.println("servlet端接收到手機傳來的第二個參數爲:" + tf2Str); //對接收的參數進行處理 String return1Str = tf1Str.concat(":早上好!"); //在接收到的參數後連接字符串 String return2Str = tf2Str.concat(":晚上好!"); //發送處理後的參數給手機 DataOutputStream dos = new DataOutputStream(response.getOutputStream()); dos.writeUTF(return1Str);dos.writeUTF(return2Str); System.out.println("servlet傳遞給手機的第一個參數爲:" + return1Str); System.out.println("servlet傳遞給手機的第二個參數爲:" + return2Str); } public void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{ doGet(request,response); } public void destroy(){} //清除資源 } |
附運行結果:
J2ME Wireless Toolkit 1.04 的 KToolbar 窗口(如圖1):
(圖1)
tomcat4.0.6 的 start tomcat 窗口(如圖2):
(圖2)
手機模擬器(如圖3):
(圖3)