我想实现以下功能:
C#客户端连接到HTTP服务器并将映像下载到磁盘。
下次客户端开始检查服务器上的图像是否比磁盘上的图像更新,在这种情况下,客户端会覆盖磁盘上的图像。
对我来说,下载图像很容易,但我不确定如何检查服务器上的图像是否更新。我怎么能实现它?我想我可以检查时间戳或图像大小(或两者),但我不知道该怎么做。
答案 0 :(得分:6)
尝试If-Modified-Since
请求字段。 http://en.wikipedia.org/wiki/List_of_HTTP_header_fields
我不确定每台服务器是否完全支持它。因此,如果它不受支持并且您仍然会获得该文件(如果支持则不是304)您可以计算校验和,如果它们不同,则考虑修改文件。或者只是覆盖 - 你将永远拥有最新的版本。
答案 1 :(得分:5)
HttpWebRequest只能使用IE缓存,所以如果所有图像都在那个缓存中,并且重写文件的成本(但不必下载它)是可以接受的,你可以利用它
如果您需要自己处理,那么:
假设:
string uri; //URI of the image.
DateTime? lastMod; // lastModification date of image previously recorded. Null if not known yet.
string eTag; //eTag of image previously recorded. Null if not known yet.
您必须在此结尾处存储这些内容,并在开头时再次检索它们(当不是新图像时)。这取决于你,鉴于此,其余的工作:
var req = (HttpWebRequest)WebRequest.Create(uri);
if(lastMod.HasValue)
req.IfModifiedSince = lastMod.Value;//note: must be UTC, use lastMod.Value.ToUniversalTime() if you store it somewhere that converts to localtime, like SQLServer does.
if(eTag != null)
req.AddHeader("If-None-Match", eTag);
try
{
using(var rsp = (HttpWebResponse)req.GetResponse())
{
lastMod = rsp.LastModified;
if(lastMod.Year == 1)//wasn't sent. We're just going to have to download the whole thing next time to be sure.
lastMod = null;
eTag = rsp.GetResponseHeader("ETag");//will be null if absent.
using(var stm = rsp.GetResponseStream())
{
//your code to save the stream here.
}
}
}
catch(WebException we)
{
var hrsp = we.Response as HttpWebResponse;
if(hrsp != null && hrsp.StatusCode == HttpStatusCode.NotModified)
{
//unfortunately, 304 when dealt with directly (rather than letting
//the IE cache be used automatically), is treated as an error. Which is a bit of
//a nuisance, but manageable. Note that if we weren't doing this manually,
//304s would be disguised to look like 200s to our code.
//update these, because possibly only one of them was the same.
lastMod = hrsp.LastModified;
if(lastMod.Year == 1)//wasn't sent.
lastMod = null;
eTag = hrsp.GetResponseHeader("ETag");//will be null if absent.
}
else //some other exception happened!
throw; //or other handling of your choosing
}
当正确实施时,电子标签比最后修改的更可靠(注意变化的亚秒级分辨率,并且由于不同的Accept- *标题反映了不同的响应)。有些实现虽然有些错误(没有特定调整的Web场上的IIS6,带有mod-gzip的Apache),因此值得拿出与电子标签相关的代码并按日期进行操作。
编辑:如果你想进一步实现HTTP缓存,你还可以存储expires和max-age(如果它们都存在则使用后者,并且它与前者不一致)并且如果它是,则完全跳过下载早于那些值表明。我已经完成了这个并且运行良好(我有一个内存缓存的对象由各种URI返回的XML创建,如果XML是新的或没有更改,我重新使用了对象),但是它可能与您的需求无关(如果您想要比服务器建议的更新鲜,或者您总是在那个窗口之外)。
答案 2 :(得分:2)
您需要阅读RFC 2616及相关的RFC(在http://www.rfc-editor.org/cgi-bin/rfcsearch.pl搜索1616)。特别是,§13是有意义的, HTTP中的缓存,第47-62页。然后阅读相关的请求/响应标头以及您可能会收到的相关状态代码。
您可以通过HttpWebRequest
和HttpWebResponse
类访问所有标头和状态值。
但是应该注意,您可以向服务器询问您想要的任何内容:最终是服务器决定是否向您发送该URI的新表示。您可能希望使用HTTP HEAD
动词而不是其GET
动词来询问服务器有关资源的信息。
HEAD方法与GET相同,只是服务器不能返回a 响应中的message-body。 HTTP标头中包含的元信息 响应HEAD请求应该与发送的信息相同 响应GET请求。该方法可用于获得元信息 关于请求所暗示的实体而不转移实体主体本身。 此方法通常用于测试超文本链接的有效性,可访问性, 和最近的修改。