鉴于各种复杂的网络环境,笔者决定采用不同的编程接口进行下载尝试,以增加程序的可用性。
这里仅介绍使用 webclient 的方法,后续的文章会介绍其他的方法。博文中主要介绍思路和关键代码,完整的 demo 附在文末。
使用代理访问网络
很多公司的员工都是通过公司设置的代理上网的。通过代理上网主要是方便公司进行各种的管制,当然也能实现一些特殊的功能… 不过这会给我们的程序访问网络带来一些问题。
其实,webclient 中的 api 已经很智能了,比如我们创建的 httpwebrequest 对象,它自带一个 proxy 属性。也就是说,webhttprequest 默认会使用找到的代理。这很棒,也能处理很多情况了。可是如果这个默认的代理需要验证域用户的身份信息,这时使用 webhttprequest 访问网络就可能失败。此时查看 proxy. credentials 属性,发现它是 null。
从 webclient 的 api 中是可以取到系统默认的 credentials 的,只是不太清楚为什么 proxy.credentials 属性默认没有设置为这个值。我们自己设置下就可以了。
1
|
request.proxy.credentials = credentialcache.defaultcredentials;
|
但实际的网络环境可能会更复杂,需要用户来指定联网的代理,并同时指定联网所需的 credentials。写法如下:
1
2
|
myproxy = new webproxy( "proxyaddress" );
myproxy.credentials = new networkcredential(proxyusername, proxyuserpasswd, domainname);
|
克服缓存
缓存可谓无处不再,在服务器端 cdn 会有缓存,在客户端的代理层也会有缓存。所以经常出现的问题是:服务器上的文件明明更新了,还是会有一些客户下载到旧文件。我们先来处理客户端的缓存问题。
httpwebrequest 的 cachepolicy.level 属性就是设置缓存策略的,只是它的默认值是 bypasscache。我们把它改为 reload 就行了:
request.cachepolicy = new system.net.cache.requestcachepolicy(system.net.cache.requestcachelevel.reload);
接下来是服务器端的缓存问题。
现在大家好像都在使用 cdn,可在使用中经常发现 cdn 端的缓存更新有问题。在网上查了查也没有什么好的解决办法,不过倒是有一个很好的 workaround,就是在请求中添加一个随机的字符串作为参数。
1
2
3
|
random rdm = new random();
string s = rdm.next().tostring();
myurl += "?" + s;
|
需要注意的是,关于缓存,一定要使用符合当前用例的策略,且不可搞一刀切。
更友好的下载过程
使用滚动条显示下载进度,显示实时的下载速度,允许用户取消下载:
下面是下载用的核心代码,我们把它分为计算下载百分比和计算当前下载速度分别介绍。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
// 获得下载文件的长度
double contentlength = downloadmanager.getcontentlength(myhttpwebclient);
byte [] buffer = new byte [buffersize];
long downloadedlength = 0;
long currenttimespandatalength = 0;
int currentdatalength;
while ((currentdatalength = stream.read(buffer, 0, buffersize)) > 0 && ! this ._canceldownload)
{
filestream.write(buffer, 0, currentdatalength);
downloadedlength += ( long )currentdatalength;
currenttimespandatalength += ( long )currentdatalength;
int intdownloadspeed = 0;
if ( this ._downloadstopwatch.elapsedmilliseconds > 800)
{
double num5 = ( double )currenttimespandatalength / 1024.0;
double num6 = ( double ) this ._downloadstopwatch.elapsedmilliseconds / 1000.0;
double doubledownloadspeed = num5 / num6;
intdownloadspeed = ( int )math.round(doubledownloadspeed, 0);
this ._downloadstopwatch.reset();
this ._downloadstopwatch.start();
currenttimespandatalength = 0;
}
double doubledownloadpersent = 0.0;
if (contentlength > 0.0)
{
doubledownloadpersent = ( double )downloadedlength / contentlength;
}
}
|
在下载的过程中计算下载百分比
首先需要从 http 请求中获得要下载文件的长度,细节请参考本文所配 demo。
1
|
double contentlength = downloadmanager.getcontentlength(myhttpwebclient);
|
每从文件流中读取一次数据,我们知道读了多少个字节(currentdatalength),累计下来就是当前已经下载了的文件长度。
1
|
downloadedlength += ( long )currentdatalength;
|
然后做个除法就行了:
1
|
doubledownloadpersent = ( double )downloadedlength / contentlength;
|
计算实时的下载速度
对于当前的下载速度,我们计算过去的一段时间内下载下来的字节数。时间段可以使用 stopwatch 来获得,我选择的时间段要求大于 800 毫秒。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
if ( this ._downloadstopwatch.elapsedmilliseconds > 800)
{
/***********************************/
// 计算上一个时间段内的下载速度
double num5 = ( double )currenttimespandatalength / 1024.0;
double num6 = ( double ) this ._downloadstopwatch.elapsedmilliseconds / 1000.0;
double doubledownloadspeed = num5 / num6;
/***********************************/
intdownloadspeed = ( int )math.round(doubledownloadspeed, 0);
// 本次网速计算完成后重置时间计时器和数据计数器,开始下次的计算
this ._downloadstopwatch.reset();
this ._downloadstopwatch.start();
currenttimespandatalength = 0;
}
|
事实上每次计算下载速度的时间段长度是不顾定的,但这并不影响计算结果,我只要保证距离上次计算超过了 800 毫秒就行了。
允许用户取消下载
对于一个执行时间比较长的任务来说,不允许用户取消它是被深恶痛绝的!尤其是网速不太好的时候。所以我们需要给用户一个选择:可以痛快(而不是痛苦)的结束当前的旅程。
而这一切对我们来说又是那么的简单!
while ((currentdatalength = stream.read(buffer, 0, buffersize)) > 0 && !this._canceldownload){}
当从数据流中读取数据时,我们检查用户是不是按下了"取消"按钮,就是这里的 this._canceldownload 变量。如果它是 true 就结束当前的下载。
至此,把用户抱怨最多的几个点都搞定了。其实也没有增加多少代码,并且每个知识点看起来都是那么的细微。但很明显的提高了用户的使用体验。这也给我们带来了一些启发,完成主要功能可能只是工作中的一部分,另外的一些工作可能并不是那么明显,需要我们不断的体会,发觉…
demo 下载地址:webclientdemo.rar
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:http://www.cnblogs.com/sparkdev/p/6031920.html