隨著越來越多手提電話和個(gè)人數(shù)字助理開始融入到信息高速公路之上,從移動(dòng)設(shè)備上訪問Web站點(diǎn)變得越來越重要。只有你實(shí)現(xiàn)移動(dòng)設(shè)備與非移動(dòng)設(shè)備互相通信的功能后,你設(shè)計(jì)的移動(dòng)設(shè)備應(yīng)用程序才可以稱的上是真正有用的。在本文中, 我們將通過一個(gè)簡(jiǎn)單的例子來學(xué)習(xí)如何實(shí)現(xiàn)手機(jī)與Servlet的網(wǎng)絡(luò)通信。該程序經(jīng)tomcat4.0.6、J2ME的MIDP簡(jiǎn)表及SUN的J2ME的無線應(yīng)用程序開發(fā)工具包編譯、配置和測(cè)試通過,同時(shí)在手機(jī)模擬器上中文顯示正常。
一些參考書都講了J2ME的網(wǎng)絡(luò)編程,但我感覺大都有點(diǎn)雷同,并且很少有完整的手機(jī)客戶端程序以及Servlet服務(wù)器端程序,雖然手機(jī)與Servlet通信的方法有多種,但經(jīng)實(shí)踐與比較,我覺得手機(jī)和Servlet通信比較簡(jiǎn)單的方法就是DataInputStream.readUTF和DataOutputStream.writeUTF的配對(duì)使用,比如手機(jī)發(fā)送數(shù)據(jù)可以在Connection的輸出流中用DataOutputStream.writeUTF依次寫入多個(gè)參數(shù),相對(duì)應(yīng)的,Servlet打開請(qǐng)求的輸入流(request.getInputStream)用DataInputStream.readUTF來依次讀出。返回的參數(shù)也一樣,Servlet用response.getOutputStream打開輸出流,并把返回值依次寫入,手機(jī)端打開Connection的輸出流并讀出即可。至于多個(gè)參數(shù)多個(gè)返回值都是沒有問題的,只是多次使用writeUTF和readUTF罷了。讀者可以參考程序中的注釋,以及運(yùn)行結(jié)果來理解手機(jī)與Servlet是如何傳遞與返回參數(shù)的以及是如何來進(jìn)行網(wǎng)絡(luò)通信的。
手機(jī)客戶端:
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);
//設(shè)置請(qǐng)求屬性
hc.setRequestMethod(HttpConnection.POST);
//設(shè)置為POST請(qǐng)求方式,默認(rèn)的請(qǐng)求方式是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服務(wù)器之間保持"keep alive"特色。
"keep alive"特色是指在MIDlet和Web服務(wù)器間始終使用同一個(gè)HTTP連接來多次傳遞數(shù)據(jù)
(在通常情況下,HTTP是無連接的協(xié)議,每次數(shù)據(jù)傳輸完畢后都將斷開連接,
而下次傳遞數(shù)據(jù)之前將重新建立連接)
//發(fā)送請(qǐng)求參數(shù)到servlet
dos = hc.openDataOutputStream();
dos.writeUTF(tf1Str); dos.writeUTF(tf2Str);
//用于發(fā)送請(qǐng)求參數(shù)給servlet
System.out.println("手機(jī)傳遞給servlet的第一個(gè)參數(shù)為:"+ tf1Str);
//主要起調(diào)試的作用,調(diào)試的結(jié)果將顯示在WTK的控制臺(tái)中
System.out.println("手機(jī)傳遞給servlet的第一個(gè)參數(shù)為:"+ tf2Str);
dos.flush(); dos.close();
//接收servlet響應(yīng)數(shù)據(jù)
ds = new DataInputStream(hc.openInputStream());
String return1Str = dis.readUTF(); String return2Str = dis.readUTF();
System.out.println("手機(jī)接收到servlet端傳來的第一個(gè)參數(shù)為:" + return1Str);
//主要起調(diào)試的作用,調(diào)試的結(jié)果將顯示在WTK的控制臺(tái)中
System.out.println("手機(jī)接收到servlet端傳來的第二個(gè)參數(shù)為:" + return2Str);
returnForm = new Form("返回的結(jié)果");
returnForm.append(return1Str); returnForm.append("\n");
//將返回的結(jié)果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服務(wù)器:
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");
//設(shè)置響應(yīng)屬性
//接收客戶端的請(qǐng)求
InputStream is = request.getInputStream();
DataInputStream dis = new DataInputStream(is);
String tf1Str = dis.readUTF();String tf2Str = dis.readUTF();
System.out.println("servlet端接收到手機(jī)傳來的第一個(gè)參數(shù)為:" + tf1Str);
// 主要起調(diào)試的作用,調(diào)試的結(jié)果顯示在tomcat的啟動(dòng)DOS窗口中
System.out.println("servlet端接收到手機(jī)傳來的第二個(gè)參數(shù)為:" + tf2Str);
//對(duì)接收的參數(shù)進(jìn)行處理
String return1Str = tf1Str.concat(":早上好!");
//在接收到的參數(shù)后連接字符串
String return2Str = tf2Str.concat(":晚上好!");
//發(fā)送處理后的參數(shù)給手機(jī)
DataOutputStream dos = new DataOutputStream(response.getOutputStream());
dos.writeUTF(return1Str);dos.writeUTF(return2Str);
System.out.println("servlet傳遞給手機(jī)的第一個(gè)參數(shù)為:" + return1Str);
System.out.println("servlet傳遞給手機(jī)的第二個(gè)參數(shù)為:" + return2Str);
}
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws
ServletException,IOException{
doGet(request,response);
}
public void destroy(){}
//清除資源
}
|
附運(yùn)行結(jié)果:
J2ME Wireless Toolkit 1.04 的 KToolbar 窗口(如圖1):
(圖1)
tomcat4.0.6 的 start tomcat 窗口(如圖2):
(圖2)
手機(jī)模擬器(如圖3):
(圖3)