在无线J2ME设备上实现http传输

news/2024/7/4 10:09:52
随着越来越多手提电话和个人数字助理开始融入到信息高速公路之上,从移动设备上访问 Web 站点变得越来

越重要。Java开创了消费设备中小型的储存容量的先河,它是用于开发手机、传呼机及其他微型设备应用程序

的理想语言。

  在本文中,我们将学习如何从一个J2ME客户机上向服务器发送一条HTTP GET请求和一条HTTP POST

求。虽然这只是一篇探讨性质的文章,但是我还是假定读者已经熟悉JavaJ2ME,以及Java MidletsMIDP

用程序)的运作机制。我们将使用J2MEMIDP简表,并利用SUNJ2ME的无线应用程序开发工具包编译、配

置和测试我们的应用程序。对于HTTP服务器,任何WWW地址都可以被访问,但是默认时我们将使用一个简单

Java Servlet来返回我们的HTTP请求的细节。

  如何使用J2ME客户机向Web服务器和类似的支持HTTP的服务器发送HTTP请求呢?答案就是使用可在

javax.microedition.io程序包中可找到的J2ME的网络类。本文就想具体阐述这个问题。

  本文概述

  使用J2ME设计无线网络应用程序

  .发送一条超文本GET请求

  .发送一条超文本POST请求

  .使用J2ME进行无线网络编程

  Java的网络编程能力是相当健壮的。Java 2标准版( J2SE)java.iojava.net程序包中定义了100多个接口

程序,类和异常。通过这些库实现的功能是很强大的,但是这只适用于传统的计算机系统,这些计算机系统有

强大的CPU处理能力,快速的内存和持久的数据储存,但是这些在大多数的无线设备上是不现实的。因此,

J2ME定义了这些函数的子集,并提供了一套用于网络和文件访问的固定的程序包--- javax.microedition.io程序

包。由于可移动设备种类繁多,这个程序包仅仅定义了一套接口,而为每个可移动设备供应厂商留下了实际的

应用程序接口实现。这就在可移植性和设备特定特征的应用中找到了一个最佳的平衡点。

  定义在javax.microedition.io类中的抽象网络和文件输入输出框架称为通用连接框架(Generic Connection

Framework,简称GCF)。GCF定义了一套有关抽象化的内容来描述不同的通信方法。最高级的抽象被称作连

接(Connection),还声明了六个接口(四个是直接的,两个是间接的)。这七个接口就构成了J2MECLDC

的一部分,CLDC是大多数的能使用Java的无线设备使用的配置。设计这个配置的目的就是为所有的CLDC设备

(手提电话,双向传呼机,低档的PDA等等)提供公用的网络和文件输入输出能力。虽然GCF的目的是公用网络

和文件输入输出框架,但是生产商并不要求实现GCF中声明的所有的接口。有的厂家可以决定只支持socket

接,而其它的厂家可以选择只支持基于数据报的通信。为了促进跨越类似装置的可移植性,MIDP规范要求所有

MIDP设备实现HttpConnection接口。HttpConnection不是GCF的一部分,但是它是从GCF的一个接口

ContentConnection衍生出来的。我们将使用HttpConnection接口构造我们样本应用程序。

  发送一个HTTP GET请求

  这一节将重点解释程序代码,在下一节中我们将只讲述被用来发送HTTP请求并检索由服务器返回的响应通

用连接框架接口和HttpConnection接口。创建MIDP用户界面的程序代码见附录。

  我们先要定义一个方法来放用于发送HTTP GET请求的代码。因为这个方法中的有些操作有潜在的抛出

IOException的可能,所以我们将把这样的意外(exception)抛给调用方法。

public String sendHttpGet( String url ) throws IOException {
HttpConnection hcon = null;
DataInputStream dis = null;
StringBuffer message = "";
try {


  第一步是使用Connector类打开一个到服务器的连接,这是GCF的关键。我们将把这个连接强制转换为需

  第一步是使用 Connector 类打开一个到服务器的连接,这是 GCF 的关键。我们将把这个连接强制转换为需

要的类型,在本例中为HttpConnection类型。

hcon = ( HttpConnection ) Connector.open( url );


  接下来,我们得到HttpConnection上的一个DataInputStream,允许我们一个字符一个字符的读取服务器

  接下来,我们得到 HttpConnection 上的一个 DataInputStream ,允许我们一个字符一个字符的读取服务器

的响应数据。

dis = new DataInputStream( hcon.openInputStream() );


  使用DataInputStreamread ()方法,服务器响应的每个字符都被集中起来放入StringBuffer对象。

int ch;
while ( ( ch = dis.read() ) != -1 ) {
message = message.append( ( char ) ch );
}



  最后,连接对象被净空以保存资源,而信息从这个方法中返回。

} finally {
if ( hcon != null ) hcon.close();
if ( dis != null ) dis.close();
}//
结束try/finally代码段
return message.toString();
}//
结束 sendGetRequest( String )

如何发送一个HTTP POST请求

  你可以想象,发送一个HTTP POST请求的处理过程其实与发送一个GET请求非常地类似。我们将修改一个

现有命令,添加少量的新的命令,并添加一个来自通用连接框架的附加的对象和一个附加的StringBuffer对象把

POST请求体重的内容发送到服务器中。剩下的命令将保持不变。

  复制我们刚才创建的sendHttpGet()方法,把它粘贴进同一个类文件,改名为sendHttpPost() 现在,我

们将修改这个新方法来发送一个HTTP POST请求到服务器。 在方法的顶部添加两个新的变量说明。 声明一个

类型为DataOutputStream的变量和另一个String类型的变量。 我们将使用DataOutputStream对象把存在于字

符串变量中的POST请求体发送到服务器中。

DataOutputStream dos = null;
String requestBody = null;


  修改connector.open()命令包含另一个参数,指出连接将允许客户端可以通过连接在服务器上读和写。

hcon = ( HttpConnection ) Connector.open( url, Connector.READ_WRITE );


  设置HttpConnection对象使用的请求方法为POST(默认的方法是GET)。

hcon.setRequestMethod( HttpConnection.POST );


  得到一个用于现有的HTTP连接的DataOutputStream对象。

dos = hc.openDataOutputStream();


  声明一个字节数组并通过检索一个来自requestBody字符串的字节数组初始化。 然后把DataOutputStream

  声明一个字节数组并通过检索一个来自 requestBody 字符串的字节数组初始化。 然后把 DataOutputStream

的缓冲写入字节数组内。

byte[] byteRequest = requestBody.getBytes();
for( int i = 0; i < byteRequest.length; i++ ) {
dos.writeByte(byteRequest[i]);
}//
结束for( int i = 0; i < byteRequest.length; i++ )

dos.flush(); //
包含本句,在某些设被上将可能会产生不可预期的结果


  调用flush ()方法的意图是发送已经写入的数据到DataOutputStream的服务器的缓冲区中。 在某些电话

  调用 flush () 方法的意图是发送已经写入的数据到 DataOutputStream 的服务器的缓冲区中。 在某些电话

上,这个操作工作正常,在其他的电话上,它导致HTTP请求的Transfer - Encoding被设置为" chunked ",有一

些随机字符被放到请求本身的前面和后面。那又怎样处理这个问题呢?这个方法调用实际上是根本不需要的。

在接下来的一行中,服务器连接打开(通过openInputStream ()),将自动输入缓冲区。因此,你最好不要调

用缓冲区的flush()方法。这个方法其余的部分保持不变,除了DataOutputStream对象必须在finally{}语句块中关

闭。

} finally {
if ( hc != null ) hc.close();

if ( dis != null ) dis.close();

if ( dos != null ) dis.close();
}//
结束 try/finally


  这就是所有的程序代码!并请参见本文后附带的程序代码。

  随着可以使用国际互联网络和支持网络的无线设备日益的增多普及,JavaJ2ME的重要性也在不断的变

  这就是所有的程序代码 ! 并请参见本文后附带的程序代码。

  随着可以使用国际互联网络和支持网络的无线设备日益的增多普及, Java J2ME 的重要性也在不断的变

大。因为HTTP协议是当前仅有的,被所有的遵从MIDP规范的设备支持的网络协议,它也是用于开发无线网络

应用程序的最好的候选者。

  在本文中,我们探究了无线网络编程的基本结构和几个核心问题,我们看了如何调用两个最常用的HTTP

求方法:GETPOSTJ2ME仍然在它的发展初期,并且无线设备也即将得到大面积的普及。所以,所有有志

投身于无线网络编程中的开发者们将得到大展拳脚的好机会。

  附录:

/*
* HttpMidlet.java
*/
import javax.microedition.midlet.*;
import javax.microedition.lcdui.*;
import javax.microedition.io.*;
import java.io.*;


public class HttpMidlet extends MIDlet implements CommandListener {
//
使用默认的URL。用户可以从图形用户接口改变这个值
private static String defaultURL = "http://localhost:8080/test/servlet/EchoServlet";

//
MIDP 显示
private Display myDisplay = null;

//
输入URL的图形用户接口组件
private Form requestScreen;
private TextField requestField;

//
用于提交请求的图形用户接口组件
private List list;
private String[] menuItems;

//
用于显示服务器响应的图形用户接口组件
private Form resultScreen;
private StringItem resultField;

//
用于requestScreen"send"按钮
Command sendCommand;
//
用于requestScreen"exit"按钮
Command exitCommand;
//
用于requestScreen"back"按钮
Command backCommand;

public HttpMidlet(){
//
初始化图形用户接口组件
myDisplay = Display.getDisplay( this );
sendCommand = new Command( "SEND", Command.OK, 1 );
exitCommand = new Command( "EXIT", Command.OK, 1 );
backCommand = new Command( "BACK", Command.OK, 1 );

//
显示请求的URL
requestScreen = new Form( "Type in a URL:" );
requestField = new TextField( null, defaultURL, 100, TextField.URL );
requestScreen.append( requestField );
requestScreen.addCommand( sendCommand );
requestScreen.addCommand( exitCommand );
requestScreen.setCommandListener( this );

//
选择想要的HTTP请求方法
menuItems = new String[] {"GET Request", "POST Request"};
list = new List( "Select an HTTP method:", List.IMPLICIT, menuItems, null );
list.setCommandListener( this );

//
先是从服务器上收到的信息
resultScreen = new Form( "Server Response:" );
resultScreen.addCommand( backCommand );
resultScreen.setCommandListener( this );

}//
结束HttpMidlet()

public void startApp() {
myDisplay.setCurrent( requestScreen );
}//
结束 startApp()

public void commandAction( Command com, Displayable disp ) {
//
当用户点击"send"按钮
if ( com == sendCommand ) {
myDisplay.setCurrent( list );
} else if ( com == backCommand ) {
requestField.setString( defaultURL );
myDisplay.setCurrent( requestScreen );
} else if ( com == exitCommand ) {
destroyApp( true );
notifyDestroyed();
}//
结束 if ( com == sendCommand )

if ( disp == list && com == List.SELECT_COMMAND ) {

String result;

if ( list.getSelectedIndex() == 0 ) //
发送一个 GET 请求到服务器
result = sendHttpGet( requestField.getString() );
else //
发送一个 POST 请求到服务器
result = sendHttpPost( requestField.getString() );

resultField = new StringItem( null, result );
resultScreen.append( resultField );
myDisplay.setCurrent( resultScreen );
}//
结束if ( dis == list && com == List.SELECT_COMMAND )
}//
结束 commandAction( Command, Displayable )

private String sendHttpGet( String url )
{
HttpConnection hcon = null;
DataInputStream dis = null;
StringBuffer responseMessage = new StringBuffer();

try {
//
使用READ权限的标准的 HttpConnection
hcon = ( HttpConnection )Connector.open( url );

//
HttpConnection取得一个 DataInputStream
dis = new DataInputStream( hcon.openInputStream() );

//
从服务器上取回响应
int ch;
while ( ( ch = dis.read() ) != -1 ) {
responseMessage.append( (char) ch );
}//
结束while ( ( ch = dis.read() ) != -1 )
}
catch( Exception e )
{
e.printStackTrace();
responseMessage.append( "ERROR" );
} finally {
try {
if ( hcon != null ) hcon.close();
if ( dis != null ) dis.close();
} catch ( IOException ioe ) {
ioe.printStackTrace();
}//
结束try/catch
}//
结束try/catch/finally
return responseMessage.toString();
}//
结束sendHttpGet( String )

private String sendHttpPost( String url )
{
HttpConnection hcon = null;
DataInputStream dis = null;
DataOutputStream dos = null;
StringBuffer responseMessage = new StringBuffer();
//
请求体
String requeststring = "This is a POST.";

try {
//
使用读写权限的 HttpConnection
hcon = ( HttpConnection )Connector.open( url, Connector.READ_WRITE );

//
设置请求方法为POST
hcon.setRequestMethod( HttpConnection.POST );

//
取得发送请求字符串的DataOutputStream
dos = hcon.openDataOutputStream();
byte[] request_body = requeststring.getBytes();

//
发送请求字符串到服务器
for( int i = 0; i < request_body.length; i++ ) {
dos.writeByte( request_body[i] );
}//
结束 for( int i = 0; i < request_body.length; i++ )

//
取得做为接收服务器响应的DataInputStream
dis = new DataInputStream( hcon.openInputStream() );

//
从服务器上取回响应
int ch;
while( ( ch = dis.read() ) != -1 ) {
responseMessage.append( (char)ch );
}//
结束while( ( ch = dis.read() ) != -1 ) {
}
catch( Exception e )
{
e.printStackTrace();
responseMessage.append( "ERROR" );
}
finally {
//
释放输入输出流和HTTP连接
try {
if( hcon != null ) hcon.close();
if( dis != null ) dis.close();
if( dos != null ) dos.close();
} catch ( IOException ioe ) {
ioe.printStackTrace();
}//
结束try/catch
}//
结束try/catch/finally
return responseMessage.toString();
}//
结束sendHttpPost( String )

public void pauseApp() {
}//
结束pauseApp()

public void destroyApp( boolean unconditional ) {
myDisplay = null;
requestScreen = null;
requestField = null;
resultScreen = null;
resultField = null;
}//
结束 destroyApp( boolean )
}//
结束HttpMidlet

 



http://www.niftyadmin.cn/n/3652857.html

相关文章

Linux之系统安全管理

Linux 之系统安全管理一. 账号安全控制1. 系统账号清理2. 密码安全控制3. 命令历史限制4. 终端自动注销5. 使用 su 命令切换用户5.1 su 切换用途5.2 密码验证5.3 限制使用 su 命令的用户二. PAM安全认证1. su命令的安全隐患2. PAM认证原理3. PAM安全认证流程三. 使用sudo机制提…

Linux之深入理解文件系统和日志分析

Linux 之深入理解文件系统和日志分析一. inode 与 block1. 文件数据包括元信息与实际数据2. block(块)3. inode(索引节点)4. 文件数据5. 结论6. inode的大小1. inode也会消耗硬盘空间2. 查询 inode 号方法7. 模拟消耗inode 节点二. 硬链接与软连接三. 恢复误删除的文件 EXT四. …

AJAX编写的用户注册实例及技术小结

我所举的这个例子是一个企业用户注册时的一个应用&#xff0c;当用户注册时检查用户名和企业名是否可用&#xff0c;以前的做法是在旁边加一个按钮&#xff0c;点击“检查”&#xff0c;就向服务器发出请求&#xff0c;然后等待……服务器返回信息&#xff0c;继续操作。   如…

Linux中DHCP动态分配IP

Linux中DHCP动态分配IP一. 使用DHCP动态分配IP一. 使用DHCP动态分配IP 配置相应的DHCP 和DHCP 中继 XSHELL 的属性也需要修改下 IP地址 虚拟机进入修改网卡页面修改 安装 dhcp 软件包 &#xff0c;并复制文件 进入到vim dhcpd.conf 进行配置文件 &#xff0c;退出后重启 在…

AJAX in Action

像其他人一样&#xff0c;当我看到一下RIA应用&#xff0c;例如Google Maps和Google Suggest的时候我都非常惊讶。我希望知道是如何实现的。现在&#xff0c;谜底揭开了&#xff0c;那就是AJAX。这是在我花了一段时间研究AJAX之后才知晓的。这里有一个很好的例子让我们知道AJAX…

Linux中DNS域名解析

DNS域名一、DNS域名二、DNS的域名结构三、DNS服务器类型四、DNS域名正向解析五、DNS域名反向解析六、构建主从DNS域名服务器六、DNS 分离服务器一、DNS域名 DNS是"域名系统”的英文缩写。它作为将域名和IP地址相互映射的一个分布式数据库&#xff0c;能够使人更方便地访问…

AJAX将成为移动Web2.0时代首选开发平台

LCC编译器要分析下面的程序&#xff0c;大体流程是这样的。#001 #include #002 #003 int main(void)#004 {#005 int nTest1 1;#006 int nTest2 2;#007 int nTest3;#008 int i;#009 #010 nTest3 nTest1 nTest2;#011 printf("nTest3 %d/r/n",nTest3);#012 …

Linux中FTP文件传输

Linux中FTP文件传输一. FTP服务二. FTP实验1. yum 安装vsfpd2. 设置匿名用户访问的FTP服务&#xff08;最大权限&#xff09;#修改配置文件3. 匿名访问测试4. 匿名通过FTP上传和下载文件三. 修改FTP 用户权限一. FTP服务 FTP服务是用来传输文件的协议 FTP服务器默认使用TCP协…