转帖请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17483273),请尊重他人的辛勤劳动成果,谢谢!

相信大家在平常的开发中,对网络的操作用到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直接拷贝来用就行了

package com.example.webservicedemo;import java.io.IOException;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;import org.ksoap2.SoapEnvelope;
import org.ksoap2.serialization.SoapObject;
import org.ksoap2.serialization.SoapSerializationEnvelope;
import org.ksoap2.transport.HttpResponseException;
import org.ksoap2.transport.HttpTransportSE;
import org.xmlpull.v1.XmlPullParserException;import android.os.Handler;
import android.os.Message;/*** 访问WebService的工具类,* * @see http://blog.csdn.net/xiaanming* * @author xiaanming* */
public class WebServiceUtils {public static final String WEB_SERVER_URL = "http://www.webxml.com.cn/WebServices/WeatherWebService.asmx";// 含有3个线程的线程池private static final ExecutorService executorService = Executors.newFixedThreadPool(3);// 命名空间private static final String NAMESPACE = "http://WebXml.com.cn/";/*** * @param url*            WebService服务器地址* @param methodName*            WebService的调用方法名* @param properties*            WebService的参数* @param webServiceCallBack*            回调接口*/public static void callWebService(String url, final String methodName,HashMap<String, String> properties,final WebServiceCallBack webServiceCallBack) {// 创建HttpTransportSE对象,传递WebService服务器地址final HttpTransportSE httpTransportSE = new HttpTransportSE(url);// 创建SoapObject对象SoapObject soapObject = new SoapObject(NAMESPACE, methodName);// SoapObject添加参数if (properties != null) {for (Iterator<Map.Entry<String, String>> it = properties.entrySet().iterator(); it.hasNext();) {Map.Entry<String, String> entry = it.next();soapObject.addProperty(entry.getKey(), entry.getValue());}}// 实例化SoapSerializationEnvelope,传入WebService的SOAP协议的版本号final SoapSerializationEnvelope soapEnvelope = new SoapSerializationEnvelope(SoapEnvelope.VER11);// 设置是否调用的是.Net开发的WebServicesoapEnvelope.setOutputSoapObject(soapObject);soapEnvelope.dotNet = true;httpTransportSE.debug = true;// 用于子线程与主线程通信的Handlerfinal Handler mHandler = new Handler() {@Overridepublic void handleMessage(Message msg) {super.handleMessage(msg);// 将返回值回调到callBack的参数中webServiceCallBack.callBack((SoapObject) msg.obj);}};// 开启线程去访问WebServiceexecutorService.submit(new Runnable() {@Overridepublic void run() {SoapObject resultSoapObject = null;try {httpTransportSE.call(NAMESPACE + methodName, soapEnvelope);if (soapEnvelope.getResponse() != null) {// 获取服务器响应返回的SoapObjectresultSoapObject = (SoapObject) soapEnvelope.bodyIn;}} catch (HttpResponseException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (XmlPullParserException e) {e.printStackTrace();} finally {// 将获取的消息利用Handler发送到主线程mHandler.sendMessage(mHandler.obtainMessage(0,resultSoapObject));}}});}/*** * * @author xiaanming* */public interface WebServiceCallBack {public void callBack(SoapObject result);}}

我们通过调用里面的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

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent" ><ListViewandroid:id="@+id/province_list"android:layout_width="match_parent"android:layout_height="wrap_content"android:cacheColorHint="@android:color/transparent"android:fadingEdge="none" ></ListView></RelativeLayout>

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

package com.example.webservicedemo;import java.util.ArrayList;
import java.util.List;import org.ksoap2.serialization.SoapObject;import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.Toast;import com.example.webservicedemo.WebServiceUtils.WebServiceCallBack;/*** 显示天气省份的Activity* * @see http://blog.csdn.net/xiaanming* * @author xiaanming**/
public class MainActivity extends Activity {private List<String> provinceList = new ArrayList<String>();@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);init();}private void init() {final ListView mProvinceList = (ListView) findViewById(R.id.province_list);//显示进度条ProgressDialogUtils.showProgressDialog(this, "数据加载中...");//通过工具类调用WebService接口WebServiceUtils.callWebService(WebServiceUtils.WEB_SERVER_URL, "getSupportProvince", null, new WebServiceCallBack() {//WebService接口返回的数据回调到这个方法中@Overridepublic void callBack(SoapObject result) {//关闭进度条ProgressDialogUtils.dismissProgressDialog();if(result != null){provinceList = parseSoapObject(result);mProvinceList.setAdapter(new ArrayAdapter<String>(MainActivity.this, android.R.layout.simple_list_item_1, provinceList));}else{Toast.makeText(MainActivity.this, "获取WebService数据错误", Toast.LENGTH_SHORT).show();}}});mProvinceList.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> parent, View view,int position, long id) {Intent intent = new Intent(MainActivity.this, CityActivity.class);intent.putExtra("province", provinceList.get(position));startActivity(intent);}});}/*** 解析SoapObject对象* @param result* @return*/private List<String> parseSoapObject(SoapObject result){List<String> list = new ArrayList<String>();SoapObject provinceSoapObject = (SoapObject) result.getProperty("getSupportProvinceResult");if(provinceSoapObject == null) {return null;}for(int i=0; i<provinceSoapObject.getPropertyCount(); i++){list.add(provinceSoapObject.getProperty(i).toString());}return list;}}

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

package com.example.webservicedemo;import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;import org.ksoap2.serialization.SoapObject;import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.Toast;import com.example.webservicedemo.WebServiceUtils.WebServiceCallBack;/*** 显示城市的Activity* * @see http://blog.csdn.net/xiaanming* * @author xiaanming**/
public class CityActivity extends Activity {private List<String> cityStringList;@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);init();}private void init() {final ListView mCityList = (ListView) findViewById(R.id.province_list);//显示进度条ProgressDialogUtils.showProgressDialog(this, "数据加载中...");//添加参数HashMap<String, String> properties = new HashMap<String, String>();properties.put("byProvinceName", getIntent().getStringExtra("province"));WebServiceUtils.callWebService(WebServiceUtils.WEB_SERVER_URL, "getSupportCity", properties, new WebServiceCallBack() {@Overridepublic void callBack(SoapObject result) {ProgressDialogUtils.dismissProgressDialog();if(result != null){cityStringList = parseSoapObject(result);mCityList.setAdapter(new ArrayAdapter<String>(CityActivity.this, android.R.layout.simple_list_item_1, cityStringList));}else{Toast.makeText(CityActivity.this, "获取WebService数据错误", Toast.LENGTH_SHORT).show();}}});mCityList.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> parent, View view,int position, long id) {Intent intent = new Intent(CityActivity.this, WeatherActivity.class);intent.putExtra("city", cityStringList.get(position));startActivity(intent);}});}/*** 解析SoapObject对象* @param result* @return*/private List<String> parseSoapObject(SoapObject result){List<String> list = new ArrayList<String>();SoapObject provinceSoapObject = (SoapObject) result.getProperty("getSupportCityResult");for(int i=0; i<provinceSoapObject.getPropertyCount(); i++){String cityString = provinceSoapObject.getProperty(i).toString();list.add(cityString.substring(0, cityString.indexOf("(")).trim());}return list;}
}

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

<?xml version="1.0" encoding="UTF-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="fill_parent"android:layout_height="fill_parent" ><ScrollViewandroid:layout_width="fill_parent"android:layout_height="fill_parent" ><LinearLayoutandroid:layout_width="match_parent"android:layout_height="match_parent" ><TextViewandroid:id="@+id/weather"android:textColor="#336598"android:textSize="16sp"android:layout_width="match_parent"android:layout_height="match_parent" /></LinearLayout></ScrollView></RelativeLayout>

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

package com.example.webservicedemo;import java.util.HashMap;import org.ksoap2.serialization.SoapObject;import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;
import android.widget.Toast;import com.example.webservicedemo.WebServiceUtils.WebServiceCallBack;/*** 显示天气的Activity* * @see http://blog.csdn.net/xiaanming* * @author xiaanming**/
public class WeatherActivity extends Activity{@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.weather_layout);init();}private void init() {final TextView mTextWeather = (TextView) findViewById(R.id.weather);ProgressDialogUtils.showProgressDialog(this, "数据加载中...");HashMap<String, String> properties = new HashMap<String, String>();properties.put("theCityName", getIntent().getStringExtra("city"));WebServiceUtils.callWebService(WebServiceUtils.WEB_SERVER_URL, "getWeatherbyCityName", properties, new WebServiceCallBack() {@Overridepublic void callBack(SoapObject result) {ProgressDialogUtils.dismissProgressDialog();if(result != null){SoapObject detail = (SoapObject) result.getProperty("getWeatherbyCityNameResult");StringBuilder sb = new StringBuilder();for(int i=0; i<detail.getPropertyCount(); i++){sb.append(detail.getProperty(i)).append("\r\n");}mTextWeather.setText(sb.toString());}else{Toast.makeText(WeatherActivity.this, "获取WebService数据错误", Toast.LENGTH_SHORT).show();}}});}
}

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

<applicationandroid:icon="@drawable/ic_launcher"android:label="@string/app_name"android:theme="@style/AppTheme" ><activityandroid:name="com.example.webservicedemo.MainActivity"android:label="@string/title_activity_main" ><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><activity android:name=".CityActivity"/><activity android:name=".WeatherActivity"></activity></application><uses-permission android:name="android.permission.INTERNET"/>

运行结果:

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

项目源码,点击下载

Android 通过WebService进行网络编程,使用工具类轻松实现相关推荐

  1. 使用WebService进行网络编程【工具类】

    相信大家在平常的开发中,对网络的操作用到HTTP协议比较多,通过我们使用Get或者Post的方法调用一个数据接口,然后服务器给我们返回JSON格式的数据,我们解析JSON数据然后展现给用户,相信很多人 ...

  2. android网络请求线程池,利用线程池实现Android客户端的http网络数据请求工具类

    该工具类值只实现了HTTP的get方法,参考get方法可轻松实现post.put.delete等方法,下面是get方法的实现 public class SimpleHttpClient { priva ...

  3. 【转】 Android快速开发系列 10个常用工具类 -- 不错

    原文网址:http://blog.csdn.net/lmj623565791/article/details/38965311 转载请标明出处:http://blog.csdn.net/lmj6235 ...

  4. Android快速开发系列 10个常用工具类

    目录(?)[+] 转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/38965311,本文出自[张鸿洋的博客] 打开大家手上的项目,基 ...

  5. Android应用开发:网络编程-2

    网络编程 Java基础:网络编程 Uri.URL.UriMatcher.ContentUris详解 Android应用开发:网络编程1 Android应用开发:网络编程2 1. 使用HttpClien ...

  6. Android应用开发:网络编程-1

    网络编程 Java基础:网络编程 Uri.URL.UriMatcher.ContentUris详解 Android应用开发:网络编程1 Android应用开发:网络编程2 1. 请求网络图片 网络交互 ...

  7. android 软键盘工具类,Android开发之弹出软键盘工具类简单示例

    本文实例讲述了Android开发之弹出软键盘工具类.分享给大家供大家参考,具体如下: package com.maobang.imsdk.util; import android.content.Co ...

  8. android文件读取工具类,Android 下读取Assets Properties操作封装工具类

    Android 下读取Assets Properties操作封装工具类 发布时间:2018-06-03作者:laosun阅读(2081) 为了方便使用,首先创建BaseApplication类,如下所 ...

  9. Android快速开发不可或缺的11个工具类(下载)

    Android快速开发不可或缺的11个工具类(下载) 源码简介 Android快速开发不可或缺的11个辅助类,其中10个来自张鸿洋的博客,1个是我平时积攒的,复制粘贴到你的项目里,添加上包名就可以直接 ...

最新文章

  1. java抓取网页标题内容_[Java教程]java 网页页面抓取标题和正文
  2. Dalvik与JVM区别
  3. 一文看尽8篇目标检测最新论文(EfficientDet/EdgeNet/ASFF/RoIMix/SCL/EFGRNet等)
  4. php 整数 比较,php中字符串和整数比较
  5. 电脑卡顿不流畅是什么原因_为什么安卓系统用久了会卡,苹果系统却依然流畅?原因找到了!...
  6. 软件工程师的6年总结
  7. python中math库最大值_python-math库解析
  8. 【视频点播最佳实践】使用OSS SDK上传视频到点播
  9. 兴图视频指挥系统如何配置服务器,兴图新科指挥调度终端
  10. linux中几个热键
  11. 基于 EntityFramework、Autofac 的 UnitOfWork 框架(一)
  12. 冲压模板自动标注LISP_CAD AutoLisp实现自动标注序号
  13. 数学建模——正态分布检验
  14. mysql innodb数据库修复_mysql innodb异常修复
  15. ArcBlock 活动 | 亮相西雅图重量级大会 全球共识建设进入新阶段
  16. PythonStudy——列表与字典推导式 List and dictionary derivation
  17. 牛客小白月赛1 I あなたの蛙が帰っています 【卡特兰数】
  18. 网盘拉新项目的变现玩法解析!
  19. nodejs require/import导包报错以及解决方法,chalk-animation库使用
  20. 保留已经定位区域MATLAB,基于HSV肤色提取与连通区域过滤的人脸定位算法(Matlab)...

热门文章

  1. 利用大数据构建智能交通
  2. [转]jquery的ajax交付时“加载中”提示的处理方法
  3. web 开发之js---理解并解决IE的内存泄漏方式
  4. JVM和GC知识点整理
  5. vue-cli的项目文件说明
  6. jpa入门 ,引自csdn感觉写的很全面
  7. InnoDB undo tablespace使用及原理
  8. 致力于绿色环保的美国大型数据中心
  9. 开源DataBase组件:FluentMigrator
  10. C#中StreamWriter与BinaryWriter的区别兼谈编码。