相信大家在平常的开发中,对网络的操作用到HTTP协议比较多,通过我们使用Get或者Post的方法调用一个数据接口,然后服务器给我们返回JSON格式的数据,我们解析JSON数据然后展现给用户,相信很多人很喜欢服务器给我们返回JSON数据格式,因为他解析方便,也有一些JSON的解析库,例如Google提供的GSON,阿里巴巴的FastJson,不过还是推荐大家使用FastJson来解析,我自己开发中也是用FastJson来解析,FastJson的介绍http://code.alibabatech.com/wiki/display/FastJSON/Home,不过有时候我们用到WebService接口来获取数据,  WebService是一种基于SOAP协议的远程调用标准,通过webservice可以将不同操作系统平台、不同语言、不同技术整合到一块。在Android SDK中并没有提供调用WebService的库,因此,需要使用第三方的SDK来调用WebService。PC版本的WEbservice客户端库非常丰富,例如Axis2,CXF等,但这些开发包对于Android系统过于庞大,也未必很容易移植到Android系统中。因此,这些开发包并不是在我们的考虑范围内。适合手机的WebService客户端的SDK有一些,比较常用的有Ksoap2,可以从http://code.google.com/p/ksoap2-android/wiki/HowToUse?tm=2进行下载,将jar包加入到libs目录下就行了,接下来带大家来调用WebService接口

首先我们新建一个工程,取名WebServiceDemo,我们从http://www.webxml.com.cn/zh_cn/web_services.aspx来获取WebService接口,这里面有一些免费的WebService接口,我们就用里面的天气接口吧http://www.webxml.com.cn/WebServices/WeatherWebService.asmx

我们新建一个WebService的工具类,用于对WebService接口的调用,以后遇到调用WebService直接拷贝来用就行了

  1. package com.example.webservicedemo;
  2.  
  3. import java.io.IOException;
  4. import java.util.HashMap;
  5. import java.util.Iterator;
  6. import java.util.Map;
  7. import java.util.concurrent.ExecutorService;
  8. import java.util.concurrent.Executors;
  9.  
  10. import org.ksoap2.SoapEnvelope;
  11. import org.ksoap2.serialization.SoapObject;
  12. import org.ksoap2.serialization.SoapSerializationEnvelope;
  13. import org.ksoap2.transport.HttpResponseException;
  14. import org.ksoap2.transport.HttpTransportSE;
  15. import org.xmlpull.v1.XmlPullParserException;
  16.  
  17. import android.os.Handler;
  18. import android.os.Message;
  19.  
  20. /**
  21. * 访问WebService的工具类,
  22. *
  23. * @see http://blog.csdn.net/xiaanming
  24. *
  25. * @author xiaanming
  26. *
  27. */
  28. public class WebServiceUtils {
  29. public static final String WEB_SERVER_URL = "http://www.webxml.com.cn/WebServices/WeatherWebService.asmx";
  30.  
  31. // 含有3个线程的线程池
  32. private static final ExecutorService executorService = Executors
  33. .newFixedThreadPool(3);
  34.  
  35. // 命名空间
  36. private static final String NAMESPACE = "http://WebXml.com.cn/";
  37.  
  38. /**
  39. *
  40. * @param url
  41. * WebService服务器地址
  42. * @param methodName
  43. * WebService的调用方法名
  44. * @param properties
  45. * WebService的参数
  46. * @param webServiceCallBack
  47. * 回调接口
  48. */
  49. public static void callWebService(String url, final String methodName,
  50. HashMap<String, String> properties,
  51. final WebServiceCallBack webServiceCallBack) {
  52. // 创建HttpTransportSE对象,传递WebService服务器地址
  53. final HttpTransportSE httpTransportSE = new HttpTransportSE(url);
  54. // 创建SoapObject对象
  55. SoapObject soapObject = new SoapObject(NAMESPACE, methodName);
  56.  
  57. // SoapObject添加参数
  58. if (properties != null) {
  59. for (Iterator<Map.Entry<String, String>> it = properties.entrySet()
  60. .iterator(); it.hasNext();) {
  61. Map.Entry<String, String> entry = it.next();
  62. soapObject.addProperty(entry.getKey(), entry.getValue());
  63. }
  64. }
  65.  
  66. // 实例化SoapSerializationEnvelope,传入WebService的SOAP协议的版本号
  67. final SoapSerializationEnvelope soapEnvelope = new SoapSerializationEnvelope(
  68. SoapEnvelope.VER11);
  69. // 设置是否调用的是.Net开发的WebService
  70. soapEnvelope.setOutputSoapObject(soapObject);
  71. soapEnvelope.dotNet = true;
  72. httpTransportSE.debug = true;
  73.  
  74. // 用于子线程与主线程通信的Handler
  75. final Handler mHandler = new Handler() {
  76.  
  77. @Override
  78. public void handleMessage(Message msg) {
  79. super.handleMessage(msg);
  80. // 将返回值回调到callBack的参数中
  81. webServiceCallBack.callBack((SoapObject) msg.obj);
  82. }
  83.  
  84. };
  85.  
  86. // 开启线程去访问WebService
  87. executorService.submit(new Runnable() {
  88.  
  89. @Override
  90. public void run() {
  91. SoapObject resultSoapObject = null;
  92. try {
  93. httpTransportSE.call(NAMESPACE + methodName, soapEnvelope);
  94. if (soapEnvelope.getResponse() != null) {
  95. // 获取服务器响应返回的SoapObject
  96. resultSoapObject = (SoapObject) soapEnvelope.bodyIn;
  97. }
  98. } catch (HttpResponseException e) {
  99. e.printStackTrace();
  100. } catch (IOException e) {
  101. e.printStackTrace();
  102. } catch (XmlPullParserException e) {
  103. e.printStackTrace();
  104. } finally {
  105. // 将获取的消息利用Handler发送到主线程
  106. mHandler.sendMessage(mHandler.obtainMessage(0,
  107. resultSoapObject));
  108. }
  109. }
  110. });
  111. }
  112.  
  113. /**
  114. *
  115. *
  116. * @author xiaanming
  117. *
  118. */
  119. public interface WebServiceCallBack {
  120. public void callBack(SoapObject result);
  121. }
  122.  
  123. }

我们通过调用里面的callWebService(String url, final String methodName,HashMap<String, String> properties,final WebServiceCallBack webServiceCallBack)就可以来获取我们想要的数据,现在讲解下里面的实现思路

  • 创建HttpTransportsSE对象。通过HttpTransportsSE类的构造方法可以指定WebService的WSDL文档的URL
  • 创建SoapObject对象,里面的参数分别是WebService的命名空间和调用方法名
  • 设置调用方法的参数值,如果没有参数,就不设置,有参数的话调用SoapObject对象的addProperty(String name, Object value)方法将参数加入到SoapObject对象中
  • 实例化SoapSerializationEnvelope,传入WebService的SOAP协议的版本号,将上面的SoapObject对象通过setOutputSoapObject(Object soapObject)设置到里面,并设置是否调用的是.Net开发的WebService和是否debug等信息
  • 因为涉及到网络操作,所以我们使用了线程池来异步操作调用WebService接口,我们在线程中调用HttpTransportsSE对象的call(String soapAction, SoapEnvelope envelope)方法就能实现对WebService的调用,并且通过soapEnvelope.bodyIn获取WebService返回的信息,但是返回的信息是在子线程中,我们需要利用Handler来实现子线程与主线程进行转换,然后在Handler的handleMessage(Message msg)中将结果回调到callBack的参数中,总体思路就是这个样子,接下来我们来使用这个工具类吧

我们先用一个ListView来显示所有的省份,然后点击每个省进去到市。市也用一个ListView来显示,最后点击市用TextView来显示获取的WebService天气情况,思路很简单

用来显示省份的布局,里面只有一个ListView

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent" >
  5.  
  6. <ListView
  7. android:id="@+id/province_list"
  8. android:layout_width="match_parent"
  9. android:layout_height="wrap_content"
  10. android:cacheColorHint="@android:color/transparent"
  11. android:fadingEdge="none" >
  12. </ListView>
  13.  
  14. </RelativeLayout>

接下来就是Activity的代码,先用工具类调用WebService方法,然后在回调方法callBack(SoapObject result)中解析数据到一个List<String>中,在设置ListView的适配器

  1. package com.example.webservicedemo;
  2.  
  3. import java.util.ArrayList;
  4. import java.util.List;
  5.  
  6. import org.ksoap2.serialization.SoapObject;
  7.  
  8. import android.app.Activity;
  9. import android.content.Intent;
  10. import android.os.Bundle;
  11. import android.view.View;
  12. import android.widget.AdapterView;
  13. import android.widget.AdapterView.OnItemClickListener;
  14. import android.widget.ArrayAdapter;
  15. import android.widget.ListView;
  16. import android.widget.Toast;
  17.  
  18. import com.example.webservicedemo.WebServiceUtils.WebServiceCallBack;
  19.  
  20. /**
  21. * 显示天气省份的Activity
  22. *
  23. * @see http://blog.csdn.net/xiaanming
  24. *
  25. * @author xiaanming
  26. *
  27. */
  28. public class MainActivity extends Activity {
  29. private List<String> provinceList = new ArrayList<String>();
  30.  
  31. @Override
  32. public void onCreate(Bundle savedInstanceState) {
  33. super.onCreate(savedInstanceState);
  34. setContentView(R.layout.activity_main);
  35. init();
  36. }
  37.  
  38. private void init() {
  39. final ListView mProvinceList = (ListView) findViewById(R.id.province_list);
  40.  
  41. //显示进度条
  42. ProgressDialogUtils.showProgressDialog(this, "数据加载中...");
  43.  
  44. //通过工具类调用WebService接口
  45. WebServiceUtils.callWebService(WebServiceUtils.WEB_SERVER_URL, "getSupportProvince", null, new WebServiceCallBack() {
  46.  
  47. //WebService接口返回的数据回调到这个方法中
  48. @Override
  49. public void callBack(SoapObject result) {
  50. //关闭进度条
  51. ProgressDialogUtils.dismissProgressDialog();
  52. if(result != null){
  53. provinceList = parseSoapObject(result);
  54. mProvinceList.setAdapter(new ArrayAdapter<String>(MainActivity.this, android.R.layout.simple_list_item_1, provinceList));
  55. }else{
  56. Toast.makeText(MainActivity.this, "获取WebService数据错误", Toast.LENGTH_SHORT).show();
  57. }
  58. }
  59. });
  60.  
  61. mProvinceList.setOnItemClickListener(new OnItemClickListener() {
  62.  
  63. @Override
  64. public void onItemClick(AdapterView<?> parent, View view,
  65. int position, long id) {
  66. Intent intent = new Intent(MainActivity.this, CityActivity.class);
  67. intent.putExtra("province", provinceList.get(position));
  68. startActivity(intent);
  69.  
  70. }
  71. });
  72.  
  73. }
  74.  
  75. /**
  76. * 解析SoapObject对象
  77. * @param result
  78. * @return
  79. */
  80. private List<String> parseSoapObject(SoapObject result){
  81. List<String> list = new ArrayList<String>();
  82. SoapObject provinceSoapObject = (SoapObject) result.getProperty("getSupportProvinceResult");
  83. if(provinceSoapObject == null) {
  84. return null;
  85. }
  86. for(int i=0; i<provinceSoapObject.getPropertyCount(); i++){
  87. list.add(provinceSoapObject.getProperty(i).toString());
  88. }
  89.  
  90. return list;
  91. }
  92.  
  93. }

点击省份进入该省份下面的市。也用一个ListView来显示市的数据,布局跟上面一样,Activity里面的代码也差不多相似,我就不过多说明了,直接看代码

  1. package com.example.webservicedemo;
  2.  
  3. import java.util.ArrayList;
  4. import java.util.HashMap;
  5. import java.util.List;
  6.  
  7. import org.ksoap2.serialization.SoapObject;
  8.  
  9. import android.app.Activity;
  10. import android.content.Intent;
  11. import android.os.Bundle;
  12. import android.view.View;
  13. import android.widget.AdapterView;
  14. import android.widget.AdapterView.OnItemClickListener;
  15. import android.widget.ArrayAdapter;
  16. import android.widget.ListView;
  17. import android.widget.Toast;
  18.  
  19. import com.example.webservicedemo.WebServiceUtils.WebServiceCallBack;
  20.  
  21. /**
  22. * 显示城市的Activity
  23. *
  24. * @see http://blog.csdn.net/xiaanming
  25. *
  26. * @author xiaanming
  27. *
  28. */
  29. public class CityActivity extends Activity {
  30. private List<String> cityStringList;
  31.  
  32. @Override
  33. public void onCreate(Bundle savedInstanceState) {
  34. super.onCreate(savedInstanceState);
  35. setContentView(R.layout.activity_main);
  36. init();
  37. }
  38.  
  39. private void init() {
  40. final ListView mCityList = (ListView) findViewById(R.id.province_list);
  41.  
  42. //显示进度条
  43. ProgressDialogUtils.showProgressDialog(this, "数据加载中...");
  44.  
  45. //添加参数
  46. HashMap<String, String> properties = new HashMap<String, String>();
  47. properties.put("byProvinceName", getIntent().getStringExtra("province"));
  48.  
  49. WebServiceUtils.callWebService(WebServiceUtils.WEB_SERVER_URL, "getSupportCity", properties, new WebServiceCallBack() {
  50.  
  51. @Override
  52. public void callBack(SoapObject result) {
  53. ProgressDialogUtils.dismissProgressDialog();
  54. if(result != null){
  55. cityStringList = parseSoapObject(result);
  56. mCityList.setAdapter(new ArrayAdapter<String>(CityActivity.this, android.R.layout.simple_list_item_1, cityStringList));
  57. }else{
  58. Toast.makeText(CityActivity.this, "获取WebService数据错误", Toast.LENGTH_SHORT).show();
  59. }
  60. }
  61. });
  62.  
  63. mCityList.setOnItemClickListener(new OnItemClickListener() {
  64.  
  65. @Override
  66. public void onItemClick(AdapterView<?> parent, View view,
  67. int position, long id) {
  68. Intent intent = new Intent(CityActivity.this, WeatherActivity.class);
  69. intent.putExtra("city", cityStringList.get(position));
  70. startActivity(intent);
  71. }
  72. });
  73. }
  74.  
  75. /**
  76. * 解析SoapObject对象
  77. * @param result
  78. * @return
  79. */
  80. private List<String> parseSoapObject(SoapObject result){
  81. List<String> list = new ArrayList<String>();
  82. SoapObject provinceSoapObject = (SoapObject) result.getProperty("getSupportCityResult");
  83. for(int i=0; i<provinceSoapObject.getPropertyCount(); i++){
  84. String cityString = provinceSoapObject.getProperty(i).toString();
  85. list.add(cityString.substring(0, cityString.indexOf("(")).trim());
  86. }
  87.  
  88. return list;
  89. }
  90. }

接下来就是点击相对应的城市调用WebService接口来获取该城市下面的天气详情啦,为了简单起见,我用一个TextView来显示天气信息,因为天气信息很多,一个屏幕显示不完,所以我们考虑在外面加一个ScrollView来进行滚动

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"
  4. android:layout_height="fill_parent" >
  5.  
  6. <ScrollView
  7. android:layout_width="fill_parent"
  8. android:layout_height="fill_parent" >
  9.  
  10. <LinearLayout
  11. android:layout_width="match_parent"
  12. android:layout_height="match_parent" >
  13.  
  14. <TextView
  15. android:id="@+id/weather"
  16. android:textColor="#336598"
  17. android:textSize="16sp"
  18. android:layout_width="match_parent"
  19. android:layout_height="match_parent" />
  20. </LinearLayout>
  21. </ScrollView>
  22.  
  23. </RelativeLayout>

Activity的代码就不做过多说明,跟上面的大同小异

  1. package com.example.webservicedemo;
  2.  
  3. import java.util.HashMap;
  4.  
  5. import org.ksoap2.serialization.SoapObject;
  6.  
  7. import android.app.Activity;
  8. import android.os.Bundle;
  9. import android.widget.TextView;
  10. import android.widget.Toast;
  11.  
  12. import com.example.webservicedemo.WebServiceUtils.WebServiceCallBack;
  13.  
  14. /**
  15. * 显示天气的Activity
  16. *
  17. * @see http://blog.csdn.net/xiaanming
  18. *
  19. * @author xiaanming
  20. *
  21. */
  22. public class WeatherActivity extends Activity{
  23. @Override
  24. public void onCreate(Bundle savedInstanceState) {
  25. super.onCreate(savedInstanceState);
  26. setContentView(R.layout.weather_layout);
  27. init();
  28. }
  29.  
  30. private void init() {
  31. final TextView mTextWeather = (TextView) findViewById(R.id.weather);
  32. ProgressDialogUtils.showProgressDialog(this, "数据加载中...");
  33. HashMap<String, String> properties = new HashMap<String, String>();
  34. properties.put("theCityName", getIntent().getStringExtra("city"));
  35.  
  36. WebServiceUtils.callWebService(WebServiceUtils.WEB_SERVER_URL, "getWeatherbyCityName", properties, new WebServiceCallBack() {
  37.  
  38. @Override
  39. public void callBack(SoapObject result) {
  40. ProgressDialogUtils.dismissProgressDialog();
  41. if(result != null){
  42. SoapObject detail = (SoapObject) result.getProperty("getWeatherbyCityNameResult");
  43. StringBuilder sb = new StringBuilder();
  44. for(int i=0; i<detail.getPropertyCount(); i++){
  45. sb.append(detail.getProperty(i)).append("\r\n");
  46. }
  47. mTextWeather.setText(sb.toString());
  48. }else{
  49. Toast.makeText(WeatherActivity.this, "获取WebService数据错误", Toast.LENGTH_SHORT).show();
  50. }
  51. }
  52. });
  53. }
  54. }

到这里我们就完成了编码工作,在运行程序之前我们需要在AndroidManifest.xml注册Activity,以及添加访问网络的权限

  1. <application
  2. android:icon="@drawable/ic_launcher"
  3. android:label="@string/app_name"
  4. android:theme="@style/AppTheme" >
  5. <activity
  6. android:name="com.example.webservicedemo.MainActivity"
  7. android:label="@string/title_activity_main" >
  8. <intent-filter>
  9. <action android:name="android.intent.action.MAIN" />
  10.  
  11. <category android:name="android.intent.category.LAUNCHER" />
  12. </intent-filter>
  13. </activity>
  14. <activity android:name=".CityActivity"/>
  15. <activity android:name=".WeatherActivity"></activity>
  16. </application>
  17.  
  18. <uses-permission android:name="android.permission.INTERNET"/>

运行结果:

省份,城市列表可以加上A-Z的排序功能,可以参考下Android实现ListView的A-Z字母排序和过滤搜索功能,实现汉字转成拼音,我这里就不添加了,需要添加的朋友自行实现,好了,今天的讲解到此结束,有疑问的朋友请在下面留言。

项目源码,点击下载

via:http://blog.csdn.net/xiaanming/article/details/17483273

从零开始学android开发-通过WebService进行网络编程,使用工具类轻松实现的更多相关文章

  1. 从零开始学android开发-通过WebService获取今日天气情况

    因为本身是在搞.NET方面的东东,现在在学习Android,所以想实现Android通过WebService接口来获取数据,网上很多例子还有有问题的.参考:Android 通过WebService进行 ...

  2. Android开发学习之路--网络编程之初体验

    一般手机都是需要上网的,一般我们的浏览器就是个webview.这里简单实现下下功能,先编写Android的layout布局: <?xml version="1.0" enco ...

  3. 从零开始学Python第八周:网络编程基础(socket)

    Socket网络编程 一,Socket编程 (1)Socket方法介绍 Socket是网络编程的一个抽象概念.通常我们用一个Socket表示"打开了一个网络链接",而打开一个Soc ...

  4. android开发时间和日期的代码实现工具类(一)

    android开发时间和日期工具类的代码实现: package com.gzcivil.utils; import android.annotation.SuppressLint; import an ...

  5. Android开发之使用Handler封装下载图片工具类(源码分享)

    假设每下载一张图片,就得重写一次Http协议,多线程的启动和handler的信息传递就显得太麻烦了,我们直接来封装一个工具类,便于我们以后在开发时随时能够调用. (1)在清单文件加入权限 <us ...

  6. Android开发学习之路-Palette颜色提取工具类使用

    视频(要FQ):https://www.youtube.com/watch?v=5u0dtzXL3PQ Palette是一个在support-v7包中的一个颜色提取工具类,用法比较简单,而且是谷歌官方 ...

  7. Android开发之封装log打印日志的工具类,实用logutils详细代码

    public final class LogUtil { /** all Log print on-off */ private final static boolean all = true; /* ...

  8. 从零开始学android开发- 应用程序窗体显示状态操作requestWindowFeature

    我们在开发程序是经常会需要软件全屏显示.自定义标题(使用按钮等控件)和其他的需求,今天这一讲就是如何控制Android应用程序的窗体显示. 首先介绍一个重要方法那就是requestWindowFeat ...

  9. 从零开始学android开发-创建第一个android项目

    打开ADT开发工具

随机推荐

  1. (1)quartz集群调度机制调研及源码分析---转载

    quartz2.2.1集群调度机制调研及源码分析 原文地址:http://demo.netfoucs.com/gklifg/article/details/27090179 引言quartz集群架构调 ...

  2. 【已解决】Android ADT中增大AVD内存后无法启动:emulator failed to allocate memory 8

    [问题] 折腾: [已解决]Android ADT中增大AVD内存后无法启动:emulator failed to allocate memory 8 过程中,增大对应AVD的内存为2G后,结果无法启 ...

  3. Ejabberd源码解析前奏--概述

    一.绪论    Ejabberd是一个用Erlang/OTP写的开源即时通讯服务器,其是跨平台.分布式.容错且基于开放标准的实时通讯系统.Ejabberd是一个功能丰富的XMPP服务器,同时适合小规模 ...

  4. fastdb中的位图应用

    位图内存管理: 每块内存用一个二进制位表示它的使用状态,如果该块内存被占用,则把对应位图中的对应位置1,如果空闲则置0,原理十分简单.计算机里面处理的位数最少的变量是字节(byte),所以也就是8位做 ...

  5. Oracle OCI-22053:溢出错误解决方法

    原文 Oracle OCI-22053:溢出错误解决方法 Oracle 数值数据类型最多可存储 38 个字节的精度.当将 Oracle 数值转换为公共语言运行库数据类型时,小数点后边的位数可能过多,这 ...

  6. NEUOJ711 异星工厂 字典树+贪心

    题意:你可以收集两个不相交区间的权值,区间权值是区间异或,问这两个权值和最大是多少 分析:很多有关异或求最大的题都是利用01字典树进行贪心,做这个题的时候我都忘了...最后是看别人代码的时候才想起来这 ...

  7. Multiple View Geometry in Computer Vision Second Edition by Richard Hartley 读书笔记(二)

    // Chapter 2介绍的是2d下的投影变换,摘录下了以下定理 Result 2.1. The point x lies on the line l if and only if xTl = 0. ...

  8. redis集群的搭建

    1.首先下载好软件包 #cd /opt/tzr/ #wget http://redis.googlecode.com/files/redis-2.6.11.tar.gz #mkdir /opt/tzr ...

  9. Multiclass Classification

    之前我们都是在Binary classification的基础上学习算法和知识. 如何使用Binary classification算法进行Multiclass classification呢? (一 ...

  10. Tkinter教程之Text(2)篇

    本文转载自:http://blog.csdn.net/jcodeer/article/details/1811347 '''Tkinter教程之Text(2)篇''''''6.使用tag来指定文本的属 ...