最近项目中采用Apache CXF 的REST 方式发布WebService实现,Android手机后台服务的开发,以下以简单是实例实现。
在项目中采用Android+REST WebService服务方式开发的手机平台很少采用 soap协议这种方式,主要soap协议解析问题,增加了代码量。 采用RESTFull 方式开发WebService的好处,相对SOAP协议的WebService来说,比较简单。同时简化了在手机解析工作,减轻了手机端的压力,提高了手机响应的效率。
手机后台服务:
- package com.easyway.rest.ws;
- import javax.ws.rs.GET;
- import javax.ws.rs.Path;
- import javax.ws.rs.Produces;
- import javax.ws.rs.core.MediaType;
- /**
- * 服务端发布一个简单的WebService服务
- * 在手机端接受服务端发送的信息.
- * 使用 Apache HttpClient 库访问 JAX-RS web 服务。Jersey 是 JAX-RS
- * 的参考实现,它简化了 Java™ 环境下的 RESTful Web 服务的开发。Android
- * 是一款流行的智能手机,本文将展示如何为 Android 创建一个 JAX-RS 客户端。
- * 您将创建一个访问 JAX-RS Web 服务的 Apache HttpClient 库客户端。
- * JAX-RS必须的jar:
- * jersey-bundle-1.8.jar,jersey-server-1.10.jar,jsr311-api-1.1.1.jar
- * asm-3.1.jar
- * 使用一个 root 资源类创建一个 RESTful Web 服务资源。root 资源类是带有 @PATH
- * 注释的 POJO。它包含至少一个带注释的方法,该注释为 @PATH、@GET、@PUT、@POST
- * 或 @DELETE。
- *
- * 在服务器上,按照 web.xml 的指定,init 参数 com.sun.jersey.config.property.resourceConfigClass
- * 作为 com.sun.jersey.api.core.PackagesResourceConfig 启动,而 init 参数
- * com.sun.jersey.config.property.packages 作为 com.easyway.rest.ws 启动。
- * 找到 root 资源类 com.easyway.rest.ws.HelloWorldResource。
- *
- * 备注:如果采用jersey发布JAXRS服务需要配置:
- * <pre>
- * <servlet>
- * <description>JAX-RS</description>
- * <servlet-name>JAX-RS-Servlet</servlet-name>
- * <servlet-class>com.sun.jersey.spi.container.servlet.ServletContainer</servlet-class>
- * <init-param>
- * <param-name>com.sun.jersey.config.property.resourceConfigClass</param-name>
- * <param-value>com.sun.jersey.api.core.PackagesResourceConfig</param-value>
- * </init-param>
- * <init-param>
- * <param-name>com.sun.jersey.config.property.packages</param-name>
- * <param-value>com.easyway.rest.ws</param-value>
- * </init-param>
- * <load-on-startup>1</load-on-startup>
- * </servlet>
- * <servlet-mapping>
- * <servlet-name>JAX-RS-Servlet</servlet-name>
- * <url-pattern>/services/*</url-pattern>
- * </servlet-mapping>
- *
- * </pre>
- *
- * @author longgangbai
- *
- */
- @Path("/helloworld")
- public class HelloWorldResource {
- /**
- * 一个简单的文本信息
- * @return
- */
- @GET
- @Produces(MediaType.TEXT_PLAIN)
- public String getClichedMessage() {
- return "Hello Android";
- }
- }
web.xml配置如下:
- <?xml version="1.0" encoding="UTF-8"?>
- <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
- <display-name>JAXRSWebService</display-name>
- <servlet>
- <description>JAX-RS</description>
- <servlet-name>JAX-RS-Servlet</servlet-name>
- <servlet-class>com.sun.jersey.spi.container.servlet.ServletContainer</servlet-class>
- <init-param>
- <param-name>com.sun.jersey.config.property.resourceConfigClass</param-name>
- <param-value>com.sun.jersey.api.core.PackagesResourceConfig</param-value>
- </init-param>
- <init-param>
- <param-name>com.sun.jersey.config.property.packages</param-name>
- <param-value>com.easyway.rest.ws</param-value>
- </init-param>
- <load-on-startup>1</load-on-startup>
- </servlet>
- <servlet-mapping>
- <servlet-name>JAX-RS-Servlet</servlet-name>
- <url-pattern>/services/*</url-pattern>
- </servlet-mapping>
- </web-app>
- 注意如果没有 Resource 将会报如下异常:
- com.sun.jersey.api.container.ContainerException: The ResourceConfig instance does not contain any root resource classes.
手机前台服务:
- package com.easyway.rest.ws;
- import java.io.IOException;
- import org.apache.http.HttpEntity;
- import org.apache.http.HttpResponse;
- import org.apache.http.client.ClientProtocolException;
- import org.apache.http.client.HttpClient;
- import org.apache.http.client.methods.HttpGet;
- import org.apache.http.impl.client.DefaultHttpClient;
- import org.apache.http.util.EntityUtils;
- import android.app.Activity;
- import android.os.Bundle;
- import android.os.StrictMode;
- import android.widget.TextView;
- /**
- * Android平台主要提供了四种数据存储方式:Shared Preferences、文件存储、Sqlite存储和网络存储。其中:
- 1)Shared Preferences 一个轻量级的键-值存储机制,专门用于存储键-值对数据,并且仅可以存储基本的数据类型
- (boolean、int、long、float和String);通常使用它来存储应用程序的配置信息。
- 2)文件存储 通过FileInputStream和FileOutputStream对文件进行操作,在Android中,文件是一个应用程序私有的,
- 一个应用程序无法读写其它应用程序的文件。
- 3)SQLite存储 SQLite是一款轻型的数据库,支持标准SQL。它的设计目标是嵌入式的,占用资源非常的低,在嵌入式设备中,
- 只需要几百K的内存就够了。Android平台也为我们提供了SQLite数据库。
- 4)网络存储 以上3种方式数据均存储在手机上,而网络存储的数据是存储在远程服务器上,手机客户端通过联接到网络来存储和获取数据。
- 今天要讲解的HttpClient正是常用的网络存储工具之一。记得最早接触HttpClient是在两年前,当时要做一个垂直搜索引擎,
- 数据自然是来源于互联网,通过一个爬虫系统不断从指定网站上爬取感兴趣的数据,然后通过Lucene搜索引擎框架实现海量数据
- 的快速检索。而爬虫系统最开始是想采用开源的爬虫框架Heritrix来实现,但接触一段时间后发现Heritrix过于庞大,而且是作
- 为一个独立的系统运行,不方便嵌入到现有的系统中,再加上学习成本高,最后还是选择了“HttpClient + HtmlParser”来实现的
- 小型爬虫系统;其中HttpClient可以模拟HTTP的POST和GET请求,用于从指定网站获取网页数据,而HtmlParser用于解析爬取到
- 的页面,过滤HTML标记,取得最终数据。
- 是不是发现HttpClient还挺强大的?让我们看看它是什么来头。"HttpClient 是 Apache Jakarta Common 下的子项目,可以用来
- 提供高效的、最新的、功能丰富的支持 HTTP 协议的客户端编程工具包,并且它支持 HTTP 协议最新的版本和建议"。如果你以前没
- 有接触过HttpClient,那么你只需要简单记住两点就可以了:
- 1)HttpClient是一个HTTP协议开发包;
- 2)HttpClient不是Android的专利。
- HttpClient的功能介绍:
- 1)实现了HTTP请求的所有方法(如GET、POST、PUT、HEAD 等);
- 2)支持自动转向;
- 3)支持 HTTPS 协议;
- 4)支持代理服务器等
- HttpClient的基本使用(以POST请求为例):
- 1)创建HttpClient实例(类似于浏览器客户端);
- HttpClient client = new DefaultHttpClient();
- 2)创建HttpPost请求,需要向HttpPost的构造方法传入所请求的URL;
- HttpPost post = new HttpPost(requestUrl);
- 3)发出POST请求(调用HttpClient的execute()方法,execute()的参数为HttpPost实例);
- HttpResponse response = client.execute(post);
- 4)读取返回结果;
- 5)释放连接;
- 6)对返回的结果进行处理。
- 在Android平台上使用HttpClient,并不需要添加额外的jar包,因为Android平台吸收了许多优秀的开源框架,其中就包括HttpClient,
- 下面就来看一个Android平台使用HttpClient的例子。
- 备注:是不是发现HttpClient很容易使用呢?其实,上面所讲解的只是HttpClient最基本的功能(发起POST请求);我们在浏览器客户端
- 所执行的大多数操作HttpClient都能够模拟,例如:提交表单、查询数据、上传下载文档、页面跳转、Session存储等。比如大家经
- 常玩“抢车位”、“偷菜”,就可以通过HttpClient编程自动实现。
- *
- *
- * 客户端通过Apache HttpClient调用JAXRS WebService的服务。
- * 为 Android 开发访问 JAX-RS Web 服务的 Apache HttpClient 客户端.
- *
- * 备注:在访问本机的JAXRS Web服务的时候不能使用localhost或者127.0.0.1,
- * 因为android模拟机会调用自身的linux内核操作系统,所以可能找不到相关的服务。
- * 最好填写ip地址如下:
- * "http://192.168.134.1:8080/JAXRSWebService/services/helloworld";
- *
- * @author longgangbai
- *
- */
- public class AndroidJAXRSWebServiceActivity extends Activity {
- private static final String processURL="http://192.168.134.1:8080/JAXRSWebService/services/helloworld";
- private TextView txResult;
- /**
- * Called when the activity is first created.
- */
- @Override
- public void onCreate(Bundle savedInstanceState) {
- ///在Android2.2以后必须添加以下代码
- //本应用采用的Android4.0
- //设置线程的策略
- StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
- .detectDiskReads()
- .detectDiskWrites()
- .detectNetwork() // or .detectAll() for all detectable problems
- .penaltyLog()
- .build());
- //设置虚拟机的策略
- StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()
- .detectLeakedSqlLiteObjects()
- //.detectLeakedClosableObjects()
- .penaltyLog()
- .penaltyDeath()
- .build());
- super.onCreate(savedInstanceState);
- //设置UI布局
- setContentView(R.layout.main);
- //获取结果显示文本框
- txResult=(TextView)findViewById(R.id.tvresult);
- //获取JAXRS WebService的结果信息
- getJAXRSWebService();
- }
- /**
- * 获取JAXRS WebService的结果信息
- */
- public void getJAXRSWebService(){
- try {
- //创建一个HttpClient对象
- HttpClient httpclient = new DefaultHttpClient();
- //创建HttpGet对象
- HttpGet request=new HttpGet(processURL);
- //请求信息类型MIME每种响应类型的输出(普通文本、html 和 XML)。允许的响应类型应当匹配资源类中生成的 MIME 类型
- //资源类生成的 MIME 类型应当匹配一种可接受的 MIME 类型。如果生成的 MIME 类型和可接受的 MIME 类型不 匹配,那么将
- //生成 com.sun.jersey.api.client.UniformInterfaceException。例如,将可接受的 MIME 类型设置为 text/xml,而将
- //生成的 MIME 类型设置为 application/xml。将生成 UniformInterfaceException。
- request.addHeader("Accept","text/plain");
- //获取响应的结果
- HttpResponse response =httpclient.execute(request);
- //获取HttpEntity
- HttpEntity entity=response.getEntity();
- //获取响应的结果信息
- String result =EntityUtils.toString(entity);
- txResult.setText(result);
- } catch (ClientProtocolException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }