[Android]解决3gwap联网失败:联网请求在设置代理与直连两种方式的切换
[Android]解决3gwap联网失败:联网请求在设置代理与直连两种方式的切换
问题现象:
碰到一个问题,UI交互表现为:联通号码在3gwap网络环境下资源一直无法下载成功。
查看Log日志,打印出的信息为:ConnectTimeoutException: Connect to /10.0.0.172:80 timed out
则问题原因是:在使用了移动网关代理后出现超时,仍无法正确联网。
解决办法:
经观察,在同样的网络环境下,发现AIO缩略图片能正常接收,跟踪其代码发现该AIO缩略图在使用移动网络联网的步骤如下(感谢issaczhang):
a. 首先判断网络环境,在移动网络及使用代理的情况,则对链接设置代理,并标识使用了代理
b. 发起链接
c. 在此处catch住连接超时异常(SocketTimeoutException/ConnectException),
如果发生了连接超时异常,则将此异常标识下来,尝试重新发起链接
d. 重新发起链接,如果在使用代理的情况下仍然发生了超时异常,则不使用代理,使用直连方式
(注:反映的3gwap联接失败的情况,采用代理失败了,但直连方式则正常连接成功了)
e. 直连方式仍发生异常,则取消直连方式,循环进入步骤a.
这是为了避免如果代理的设置本身是错误的,经用户修正后,保证仍可使用最新的代理联网成功。
f. 如果网络环境切换到了新的代理,则优先使用该代理进行步骤a的尝试。
经以上步骤,3gwap连网情况下,终于将使用直连方式时正常访问到网络资源。
个人觉得移动网关毕竟是2G时代的特色产物,经此问题,发现网关已经慢慢在退出历史舞台,以后不使用代理直接连接应该是大势所趋了。
使用HttpURLConnection实现的过程中仍遇到了几个坑,现在逐一填埋一下:
1号天坑(危地马拉天坑):设置代理方式的不同导致连接假成功,返回的内容是被跳转的网页内容。
默认设置代理的方式是:
HttpURLConnection conn;
java.net.Proxy proxy = new java.net.Proxy(Type.HTTP, new InetSocketAddress(defaultHost, defaultPort));
URL proxyURL = new URL(url);
conn = (HttpURLConnection) proxyURL.openConnection(proxy);
经实践,此方式在3gwap下,连接是正常的,response_code是HTTP_OK(200),也能正常读取内容,但这时已经陷坑里了,读取到的内容却是如下:
<html>
<head>
<meta http-equiv="refresh" content="0; url=http://59.151.106.150" />
</head>
<body>
</body>
</html>
你妹呀!这是一个跳转网页,如果是浏览器的话则会观察到将会跳转到下面这个页面(见图3gwap.interrupted.web.jpg):
很明显,这是一个被移动运营商拦截的广告页。没节操呀!!
经比较AIO缩略图发现该设置代理的实现是另一种方式,设置X-Online_Host方式(感谢issac):
HttpURLConnection conn;
URL hostUrl;
String host = null;
String path = null;
int hostIndex = "http://".length();
int pathIndex = url.indexOf('/', hostIndex);// url为原始的请求链接
if (pathIndex < 0) {
host = url.substring(hostIndex);
path = "";
} else {
host = url.substring(hostIndex, pathIndex);
path = url.substring(pathIndex);
}
hostUrl = new URL("http://" + defaultHost + ":" + defaultPort + path);
conn = (HttpURLConnection) hostUrl.openConnection();
conn.setRequestProperty("X-Online-Host", host);
实践后,在3gwap联网下,使用该代理,会抛出连接超时异常,这时尝试不使用代理直接请求原始url,则正常访问到目标资源了。
目前HttpCommunicator在移动网络下,如果是cmwap/uniwap/3gwap,都会采用设置X-Online-Host的方式去代理连接。
2号坑(重庆奉节小寨天坑):读取头信息(content-length)数值偏小
经过以上的处理之后,连接成功了,但取到的头信息content-length值却偏小,异常下载成功判断失败.
偏小见下图content_length_lossy.png
经实践,在设置了头信息"Accept-Encoding, "identity",后,则content-length恢复正常。
此现象在访问imgcache.qq.com上面的静态资源时必现,但访问其它网站的静态资源,倒是不必现的。
后记:
上述是使用HttpURLConnection时遇到的情况,但为了相比较,本人又尝试着使用HttpClient/HttpGet等Apache联网组件去实现,
虽然实现过程代码稍复杂一些,并未有上述的两个问题坑。
所以在联网过程中,两者皆宜,只是HttpURLConnection要步步小心。
附Apache联网组件HttpClient/HttpGet设置代理方式:
HttpHost httpHost = new HttpHost(defaultHost, defaultPort);
para.setParameter(ConnRoutePNames.DEFAULT_PROXY, httpHost);
httpGet.setParams(para);
// ... ...
httpClient.execute(httpGet);
[Android]解决3gwap联网失败:联网请求在设置代理与直连两种方式的切换的更多相关文章
- [Swift]Alamofire:设置网络请求超时时间【timeout】的两种方式
两种方式作用相同,是同一套代码的两种表述. 第一种方式:集聚. 直接设置成员属性(全局属性),这种方法不能灵活修改网络请求超时时间timeout. 声明为成员属性: // MARK: - 设置为全局变 ...
- android启动第一个界面时即闪屏的核心代码(两种方式)
闪屏,就是SplashScreen,也能够说是启动画面,就是启动的时候,闪(展示)一下,持续数秒后.自己主动关闭. 第一种方式: android的实现很easy,使用Handler对象的postDe ...
- vue之provide和inject跨组件传递属性值失败(父组件向子组件传值的两种方式)
简单介绍:当一个子组件需要用到父组件的父组件的某些参数.那么这个时候为了避免组件重复传参,使用vue的依赖注入是个不错的方法,直接在最外层组件设置一个provide,内部不管多少嵌套都可以直接取到最外 ...
- 【Android】adb connect 手机的两种方式
adb支持两种连接Android系统的方式,USB方式及网络方式.一般android手机及android平板默认会设置为USB方式(直接插数据线的方式). 下边介绍两种方式的切换方式. 1. 背景知识 ...
- Android 应用开发 之通过AsyncTask与ThreadPool(线程池)两种方式异步加载大量数据的分析与对比--转载
在加载大量数据的时候,经常会用到异步加载,所谓异步加载,就是把耗时的工作放到子线程里执行,当数据加载完毕的时候再到主线程进行UI刷新.在数据量非常大的情况下,我们通常会使用两种技术来进行异步加载,一 ...
- Android请求服务器的两种方式--post, get的区别
android中用get和post方式向服务器提交请求_疯狂之桥_新浪博客http://blog.sina.com.cn/s/blog_a46817ff01017yxt.html Android提交数 ...
- 解决 SharePoint 2010 拒绝访问爬网内容源错误的小技巧(禁用环回请求的两种方式)
这里有一条解决在SharePoint 2010搜索爬网时遇到的“拒绝访问错误”的小技巧. 首先要检查默认内容访问帐户是否具有相应的访问权限,或者添加一条相应的爬网规则.如果目标资源库是一个ShareP ...
- 分布式理论基础(一)一致性及解决一致性的两种方式:2PC和3PC (转载 不错)
分布式理论基础(一)一致性及解决一致性的两种方式:2PC和3PC 1 一致性 1.1 简述 一致性,是指对每个节点一个数据的更新,整个集群都知道更新,并且是一致的 假设一个具有N个节点的分布式系统,当 ...
- C#中Post请求的两种方式发送参数链和Body的
POST请求 有两种方式 一种是组装key=value这种参数对的方式 一种是直接把一个字符串发送过去 作为body的方式 我们在postman中可以看到 sfdsafd sdfsdfds publi ...
随机推荐
- Google App Engine 学习和实践
这个周末玩了玩Google App Engine,随手写点东西,算是学习笔记吧.不当之处,请多多指正. 作者:liigo,2009/04/26夜,大连 原创链接:http://blog.csdn.ne ...
- uva 10692 - Huge Mods(数论)
题目链接:uva 10692 - Huge Mods 题目大意:给出一个数的次方形式,就它模掉M的值. 解题思路:依据剩余系的性质,最后一定是行成周期的,所以就有ab=abmod(phi[M])+ph ...
- 浅谈C#中的泛型
1.什么是泛型? 泛型是程序设计语言的一种特性.允许程序员在强类型程序设计语言中编写 代码时定义一些可变部分,那些部分在使用前必须作出指明.各种程序设计语言和其编译器.运行环境对泛型的支持均不一样.将 ...
- 几十篇GDI以及MFC自绘的文章
http://www.cnblogs.com/lidabo/category/434801.html
- Oracle中如何插入特殊字符:& 和 ' (多种解决方案)
今天在导入一批数据到Oracle时,碰到了一个问题:Toad提示要给一个自定义变量AMP赋值,一开始我很纳闷,数据是一系列的Insert语句,怎么会有自定义变量呢?后来搜索了一下关键字AMP发现,原来 ...
- 【COCOS2DX-游戏开发之三三】TMX边界控制与小窗体内预览TMX
做一款像素游戏,须要确定地图的边界.保证人物的位置位于屏幕中央.到达地图左边界.地图位置不变.人向左走,到达右边界,地步位置不变,人向右走 如:地图左边.右边,上边空出的边界.还有下方留出操作butt ...
- MySQL中同一时候存在创建和上次更新时间戳字段解决方法浅析
在写这篇文章之前.明白我的MySQL版本号. mysql> SELECT VERSION(); +------------+ | VERSION() | +------------+ | 5.5 ...
- 【m从翻译os文章】写日志禁令Sqlnet.log和Listener.log
写日志禁令Sqlnet.log和Listener.log 参考原始: How to Disable Logging to the Sqlnet.log and the Listener.log (Do ...
- Songs
Two Steps From Hell - Strength of a Thousand Men
- 纯C语言INI文件解析
原地址:http://blog.csdn.net/foruok/article/details/17715969 在一个跨平台( Android .Windows.Linux )项目中配置文件用 IN ...