Android_WebServices_源代码分析
在Android_WebServices_介绍一文中,简介了WebServices的基础知识。以下主要分析 ksoap2-android-assembly-3.3.0-jar-with-dependencies.jar实现源代码。
1.调用WebServices流程
public void getRemoteInfo(String phoneSec) {
String nameSpace = "http://WebXml.com.cn/";
String methodName = "getMobileCodeInfo";
String endPoint = "http://webservice.webxml.com.cn/WebServices/MobileCodeWS.asmx";
String soapAction = "http://WebXml.com.cn/getMobileCodeInfo";
// 1.初始化 SoapObject对象。为该方法设置參数。相当于信体
SoapObject request = new SoapObject(nameSpace, methodName);
request.addProperty("mobileCode", phoneSec);
request.addProperty("userId", "");
// 2.实例化SoapSerializationEnvelope对象,相当于信皮
SoapSerializationEnvelope envelope = new SoapSerializationEnvelope(SoapEnvelope.VER10);
envelope.bodyOut = request;
envelope.dotNet = true;//兼容.net开发的Net-Services
// 3.实例化HttpTransportSE对象,还能够指定放了訪问的请求时间
HttpTransportSE transport = new HttpTransportSE(endPoint);
//HttpTransportSE transport = new HttpTransportSE(endPoint, timeout);
try {
// 4.核心方法调用,当中soapActon在SoapSerializationEnvelope.VER12时无效,且为POST请求
transport.call(soapAction, envelope);
SoapObject response = (SoapObject) envelope.bodyIn;
final String result = response.getProperty(0).toString();//Vector
toast(result);
} catch (Exception e) {
e.printStackTrace();
toast(e.getMessage());
}
}
2.transport.call关键源代码分析
/**
* Perform a soap call with a given namespace and the given envelope providing
* any extra headers that the user requires such as cookies. Headers that are
* returned by the web service will be returned to the caller in the form of a
* <code>List</code> of <code>HeaderProperty</code> instances.
*
* @param soapAction
* the namespace with which to perform the call in.
* @param envelope
* the envelope the contains the information for the call.
* @param headers
* <code>List</code> of <code>HeaderProperty</code> headers to send with the SOAP request.
* @param outputFile
* a file to stream the response into rather than parsing it, streaming happens when file is not null
*
* @return Headers returned by the web service as a <code>List</code> of
* <code>HeaderProperty</code> instances.
*
* @throws HttpResponseException
* an IOException when Http response code is different from 200
*/
public List call(String soapAction, SoapEnvelope envelope, List headers, File outputFile)
throws HttpResponseException, IOException, XmlPullParserException { if (soapAction == null) {
soapAction = "\"\"";
} //依据envelope,将其序列化为一个请求字节数组
byte[] requestData = createRequestData(envelope, "UTF-8"); // debug=true, requestDump的值为请求的数据,方便调试
requestDump = debug ? new String(requestData) : null;
responseDump = null; //connection = new ServiceConnectionSE(proxy, url, timeout);包含设置时间
ServiceConnection connection = getServiceConnection(); connection.setRequestProperty("User-Agent", USER_AGENT);
// SOAPAction is not a valid header for VER12 so do not add
// it
// @see "http://code.google.com/p/ksoap2-android/issues/detail? id=67
if (envelope.version != SoapSerializationEnvelope.VER12) {
connection.setRequestProperty("SOAPAction", soapAction);
} if (envelope.version == SoapSerializationEnvelope.VER12) {
connection.setRequestProperty("Content-Type", CONTENT_TYPE_SOAP_XML_CHARSET_UTF_8);
} else {
connection.setRequestProperty("Content-Type", CONTENT_TYPE_XML_CHARSET_UTF_8);
} // this seems to cause issues so we are removing it
//connection.setRequestProperty("Connection", "close");
connection.setRequestProperty("Accept-Encoding", "gzip"); // Pass the headers provided by the user along with the call
if (headers != null) {
for (int i = 0; i < headers.size(); i++) {
HeaderProperty hp = (HeaderProperty) headers.get(i);
connection.setRequestProperty(hp.getKey(), hp.getValue());
}
} // POST请求
connection.setRequestMethod("POST");
//发送数据。耗时较长,将requestData发送至connection的输出流
sendData(requestData, connection, envelope); requestData = null;
InputStream is = null;
List retHeaders = null;
byte[] buf = null; // To allow releasing the resource after used
int contentLength = 8192; // To determine the size of the response and adjust buffer size
boolean gZippedContent = false;
boolean xmlContent = false;
// 得到响应码
int status = connection.getResponseCode(); try {
//得到响应头
retHeaders = connection.getResponseProperties(); for (int i = 0; i < retHeaders.size(); i++) {
HeaderProperty hp = (HeaderProperty)retHeaders.get(i);
// HTTP response code has null key
if (null == hp.getKey()) {
continue;
} // If we know the size of the response, we should use the size to initiate vars
if (hp.getKey().equalsIgnoreCase("content-length") ) {
if ( hp.getValue() != null ) {
try {
contentLength = Integer.parseInt( hp.getValue() );
} catch ( NumberFormatException nfe ) {
contentLength = 8192;
}
}
} // Check the content-type header to see if we're getting back XML, in case of a
// SOAP fault on 500 codes
if (hp.getKey().equalsIgnoreCase("Content-Type")
&& hp.getValue().contains("xml")) {
xmlContent = true;
} // ignoring case since users found that all smaller case is used on some server
// and even if it is wrong according to spec, we rather have it work..
if (hp.getKey().equalsIgnoreCase("Content-Encoding")
&& hp.getValue().equalsIgnoreCase("gzip")) {
gZippedContent = true;
}
} //first check the response code....
if (status != 200) {
//throw new IOException("HTTP request failed, HTTP status: " + status);
throw new HttpResponseException("HTTP request failed, HTTP status: " + status, status);
} if (contentLength > 0) {
if (gZippedContent) {
is = getUnZippedInputStream(
new BufferedInputStream(connection.openInputStream(),contentLength));
} else {
is = new BufferedInputStream(connection.openInputStream(),contentLength);
}
}
} catch (IOException e) {
if (contentLength > 0) {
if(gZippedContent) {
is = getUnZippedInputStream(
new BufferedInputStream(connection.getErrorStream(),contentLength));
} else {
is = new BufferedInputStream(connection.getErrorStream(),contentLength);
}
} if ( e instanceof HttpResponseException) {
if (!xmlContent) {
if (debug && is != null) {
//go ahead and read the error stream into the debug buffers/file if needed.
readDebug(is, contentLength, outputFile);
} //we never want to drop through to attempting to parse the HTTP error stream as a SOAP response.
connection.disconnect();
throw e;
}
}
} // debug=true responseDump=响应数据,方便调试
if (debug) {
is = readDebug(is, contentLength, outputFile);
} // 依据is流,将流数据解析至 envelope.bodyIn中去
parseResponse(envelope, is,retHeaders); //释放资源
is = null;
buf = null;
connection.disconnect();
connection = null;
// 返回响应头
return retHeaders;
}
Android_WebServices_源代码分析的更多相关文章
- android-plugmgr源代码分析
android-plugmgr是一个Android插件加载框架,它最大的特点就是对插件不需要进行任何约束.关于这个类库的介绍见作者博客,市面上也有一些插件加载框架,但是感觉没有这个好.在这篇文章中,我 ...
- Twitter Storm源代码分析之ZooKeeper中的目录结构
徐明明博客:Twitter Storm源代码分析之ZooKeeper中的目录结构 我们知道Twitter Storm的所有的状态信息都是保存在Zookeeper里面,nimbus通过在zookeepe ...
- 转:SDL2源代码分析
1:初始化(SDL_Init()) SDL简介 有关SDL的简介在<最简单的视音频播放示例7:SDL2播放RGB/YUV>以及<最简单的视音频播放示例9:SDL2播放PCM>中 ...
- 转:RTMPDump源代码分析
0: 主要函数调用分析 rtmpdump 是一个用来处理 RTMP 流媒体的开源工具包,支持 rtmp://, rtmpt://, rtmpe://, rtmpte://, and rtmps://. ...
- 转:ffdshow 源代码分析
ffdshow神奇的功能:视频播放时显示运动矢量和QP FFDShow可以称得上是全能的解码.编码器.最初FFDShow只是mpeg视频解码器,不过现在他能做到的远不止于此.它能够解码的视频格式已经远 ...
- UiAutomator源代码分析之UiAutomatorBridge框架
上一篇文章<UIAutomator源代码分析之启动和执行>我们描写叙述了uitautomator从命令行执行到载入測试用例执行測试的整个流程.过程中我们也描写叙述了UiAutomatorB ...
- MyBatis架构设计及源代码分析系列(一):MyBatis架构
如果不太熟悉MyBatis使用的请先参见MyBatis官方文档,这对理解其架构设计和源码分析有很大好处. 一.概述 MyBatis并不是一个完整的ORM框架,其官方首页是这么介绍自己 The MyBa ...
- hostapd源代码分析(三):管理帧的收发和处理
hostapd源代码分析(三):管理帧的收发和处理 原文链接:http://blog.csdn.net/qq_21949217/article/details/46004379 这篇文章我来讲解一下h ...
- hostapd源代码分析(二):hostapd的工作机制
[转]hostapd源代码分析(二):hostapd的工作机制 原文链接:http://blog.csdn.net/qq_21949217/article/details/46004433 在我的上一 ...
随机推荐
- 未能加载文件或程序集“file:///D:/Program Files (x86)/ArcGIS/DeveloperKit10.0/DotNet/ESRI.ArcGIS.3DAnalyst.dll”或它的某一个依赖项。试图加载格式不正确的程序。 行 129,位置 5。
能加载文件或程序集“file:///C:/Program Files (x86)/ArcGIS/DeveloperKit10.0/DotNet/ESRI.ArcGIS.ADF.Local.dll”或它 ...
- DCI:DCI学习总结
备注 之前把DCI的Role和四色原型的Role给弄混了,本文也不会比较这两种Role的区别(后面有机会再说),这里简单的记录一下对DCI的理解. 参考文章:http://www.cnblogs.co ...
- pytest文档10-命令行传参
前言 命令行参数是根据命令行选项将不同的值传递给测试函数,比如平常在cmd执行"pytest --html=report.html",这里面的"--html=report ...
- Scrum 思考
下个月就要离职,所以这个月特别清闲,上班时间都在上网看书,偶然在Startup News的一篇文章(http://blog.devtang.com/blog/2013/06/17/startup-an ...
- C++构造函数、析构函数、虚析构函数
1.构造函数 C++中的构造函数是用于初始化类的各种变量以及分配资源等.主要的注意事项是: (1)在继承关系中先初始化父类对象后初始化子类对象. (2)在一个类中按照变量的声明顺序,对类中的变量进行初 ...
- Linux下QTCreator代码自动补全(是真的自动补全,不是手动触发)
在使用Windows下的QTCreator的时候,像visual studio一样代码自动补全十分方便,而在Linux下,QTCreator似乎不能做到. 网上有些说是可以设置成手动补全,今天试了一下 ...
- [Functional Programming] Unbox types with foldMap
Previously we have seen how to use Concat with reduce: ), Sum(), Sum()] .reduce((acc, x) => acc.c ...
- mvc:annotation-driven注解的作用
<mvc:annotation-driven>会自动注册RequestMappingHandlerMapping与RequestMappingHandlerAdapter两个Bean,这是 ...
- 你远比想象中强大pdf
读后感: 序 一.强化自我认知 认识你自己 你认为什么东西是最重要的呢? 这个问题的答案就是价值观. 让定期审视人生成为习惯 除去恐惧 树立目标 二.改变思维模式 选择,记住你的选择(做决定) 巅 ...
- 娓娓道来c指针 (4)解析c的声明语句
(4)解析c的声明语句 在继续探索c指针之前.有必要来解析下c语言中复杂的声明语法. 仅仅须要记住两则:一个原则,一个规则. 原则:先看标示符. 规则:运算符优先级是规则. 举例说明 1.最简单的 i ...