gisempire100
捉鬼专家
捉鬼专家
  • 注册日期2004-08-13
  • 发帖数552
  • QQ
  • 铜币2462枚
  • 威望0点
  • 贡献值0点
  • 银元0个
阅读:1132回复:0

在ASP.NET中缓存图像

楼主#
更多 发布于:2008-04-25 20:56
原文地址:http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET.aspx<BR><BR>一个最简单但是最有效的提高web应用程序性能的方法是:在客户端缓存图像。<BR><EM></EM><STRONG>简介</STRONG><BR>有许多方法可以提高web应用程序的性能,一个最简单但是最有效的提高web应用程序性能的方法是,在客户端缓存图像。在这篇文章中<BR>我将展示怎样给我们的DotNetNuke站点实现图片缓存。<BR><BR><STRONG>问题</STRONG><BR>当我在搭建网站http://www.software-architects.com的时候,我在制作菜单项的时候,在CSS中使用了很多的图像作为背景。在把这些文件上传到网站服务器之后,我用Microsoft Network Monitor测试下了一个请求会产生多少的流量。这个工具可以捕获和分析网络的流量。有可以从微软下载中心 <a href="http://www.microsoft.com/downloads/details.aspx?FamilyID=18b1d59d-f4d8-4213-8d17-2f6dde7d7aac;amp;amp;amp;DisplayLang=en" target="_blank" ><FONT color=#1a8bc8>(Microsoft Download Center)</FONT></A>.下载到它。<BR><BR>用Microsoft Network Monitor 3.1 我记录了对http://www.software-architects.com的一次请求。结果在一个页面中,我得到了对于20个不同的文件夹的20个不同的请求。Microsoft Network Monitor 显示大约有一半的请求是针对对菜单中的图片的。<BR><BR><IMG src="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/MNM_withoutCaching.png"><BR>有两种不同的方法避免这个问题。一种是,你可以告诉IIS在客户端缓存图片;另外一个方法是,直接在ASP.NET中做这个(这个方法更复杂些)<BR><STRONG>在IIS中缓存图像</STRONG><BR>在IIS中缓存是非常简单的。打开IIS,在左侧或者是右侧面板中选择一个文件夹,打开属性对话框。<IMG src="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/IIS_Images.png"><BR>勾上“启用内容过期(Enable content expiration)”并且选择你的内容什么时候过期。<BR><BR><IMG src="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/IIS_CachingProperties.png"><BR>这样就行了!IIS用“cache-control”告诉客户端内容可能被缓存在客户端。 "expires" 里边包含了过期的日期。所以客户端知道在这个日期之后,它必须请求服务器来或得新的内容。<BR><BR>如果符合下边这两点,这个方法将很好用:<BR>
<UL>
<LI>你可以把你所有的图片和其他要缓存的文件放到一个或者少数几个文件夹中。
<LI>最重要的是,你有权限访问IIS。 </LI></UL>在我们的案列中,这两个条件都不能满足。在我们的 DotNetNuke 工程中,图片被分布在多个文件夹中。所以配置IIS是非常复杂的。并且最重要的是,我们的空间商没有给我们访问IIS的权限。因此,我们必须寻找另外的解决方案。<STRONG><BR></STRONG><STRONG>用自定义个HttpHandler缓存图片</STRONG><BR>第一件事情,我们要解决的是绕过IIS得到ASP.NET的请求。我决定写一个自定义的HTTP handler,它可以监听"*.gif .ashx,".jpg .ashx "和“.png.ashx”这些类型的文件。你可以在这个网站找到关于IHttpHandler一篇好文章 <a href="http://microsoft.apress.com/asptodayarchive/72809/use-local-scope-to-improve-performance" target="_blank" ><FONT color=#1a8bc8>Use local scope to improve performance</FONT></A>.<BR>我在vs中用<CODE>CachingHandler这个类</CODE>建了一个类的工程,它可以处理对图片的请求。 <CODE>CachingHandler 实现了了</CODE><CODE>IHttpHandler的接口</CODE>,就像page类一样。这个接口提供了<CODE>IsReusable</CODE> 属性和<CODE>ProcessRequest方法。</CODE><BR><CODE>IsResuable</CODE> 表明另外一个请求是否可以复用这个Http handler。<BR><CODE>ProcessRequest</CODE> 是关于线程安全的 。<BR><CODE>ProcessRequest</CODE> 做真正的工作。它得到当前的内容,并对发送给客户端的结果负责<BR><PRE> namespace SoftwareArchitects.Web<BR>
<BR>
{<BR>
<BR>
public class CachingHandler : IHttpHandler<BR>
<BR>
{<BR>
<BR>
public bool IsReusable<BR>
<BR>
{<BR>
<BR>
get { return true; }<BR>
<BR>
}<BR>
<BR>
public void ProcessRequest(HttpContext context)<BR>
<BR>
{...<BR>
<BR>
}<BR>
<BR>
}<BR>
<BR>
}<BR>
<BR>
我们想让我们的HTTP handler发送个文件给客户端。当我们用这些路径 <EM>*.gif.ashx</EM>, <EM>*.jpg.ashx</EM> 和<BR>
<BR>
<EM>*.png.ashx监听文件的时候,我们所必须做到就是把”.ashx“从</EM>请求的路径中移除来得到我们想发送给客户端<BR>
<BR>
的文件。包括我们提取的文件名和这个文件的扩展名。<BR>
<PRE>public void ProcessRequest(HttpContext context)<BR>
<BR>
{<BR>
<BR>
string file = context.Server.MapPath<BR>
<BR>
(context.Request.FilePath.Replace(".ashx", ""));<BR>
<BR>
string filename = file.Substring(file.LastIndexOf('\\') + 1);<BR>
<BR>
string extension = file.Substring(file.LastIndexOf('.') + 1);</PRE>
<BR>
<BR>
<BR>
  下一步,我们载入从webconfig中为CachingHandler做的配置。所以我创建了个类CachingSection<BR>
<BR>
(我展示的有点晚了),它包含了一个属性CachingTimeSpan 和一个集合FileExtensions,它可以知道<BR>
<BR>
每个文件扩展的内容类型。在这个类的帮助下,我们可以配置HttpCachePolicy响应对象。</PRE>
<UL>
<LI><CODE>SetExpires 告诉客户端内容的有效期是多长。</CODE>
<LI><CODE>SetCacheability 告诉客户端谁可以缓存内容。我们设置这个属性是 public。它意味着这个响应可以被客户端缓存起来并且可以分享缓存。</CODE>
<LI><CODE>SetValidUnitExpires 指定ASP.NET缓存是否应该忽略客户端发送的使cookie无效的HTTP cache-control 头</CODE>
<LI><CODE>ContentType 设置响应的MIME类型</CODE> </LI></UL><PRE>CachingSection config = (CachingSection)context.GetSection<BR>
<BR>
("SoftwareArchitects/Caching");<BR>
<BR>
if (config != null)<BR>
<BR>
{<BR>
<BR>
context.Response.Cache.SetExpires<BR>
<BR>
(DateTime.Now.Add(config.CachingTimeSpan));<BR>
<BR>
context.Response.Cache.SetCacheability(HttpCacheability.Public);<BR>
<BR>
context.Response.Cache.SetValidUntilExpires(false);<BR>
<BR>
FileExtension fileExtension = config.FileExtensions[extension];<BR>
<BR>
if (fileExtension != null){<BR>
<BR>
context.Response.ContentType = fileExtension.ContentType;<BR>
<BR>
}<BR>
<BR>
}<BR>
<BR>
最后,我们对响应添加content-disposition头告诉客户端它应该在浏览器中打开文件(inline)。另外<BR>
<BR>
我们可以把文件名改成没有扩展名.ashx的名字。因为这个名字,当你尽力下载这个文件的时候才显示。然后<BR>
<BR>
我们使用writeFile把这个文件发送给客户端。<BR>
<PRE>  context.Response.AddHeader("content-disposition", <BR>
<BR>
"inline; filename=" + filename);<BR>
<BR>
context.Response.WriteFile(file);}<BR>
<BR>
<STRONG>在webconfig中定义自定义配</STRONG></PRE>
<BR>
  在http handler中我们使用一个自定义的类去读webconfig中的一些配置。因此,我创建了<CODE>CachingSection <BR>
<BR>
类,继承自</CODE><CODE>ConfigurationSection。在这个类中,我实现了一个属性</CODE><CODE>CachingTimeSpan,他存储着一个<BR>
<BR>
timespan的值,这个值是在客户端缓存对象的时间,并且一个属性</CODE><CODE>FileExtensions,它存储着</CODE><CODE>FileExtension</CODE> <BR>
<BR>
对象的集合。为了标注webcofing中这些元素的属性你只不过是在每个属性中添加一个<CODE>ConfigurationProperty</CODE> <BR>
<BR>
属性,它可以在webconfig中设置。<BR>
<PRE>namespace SoftwareArchitects.Web.Configuration<BR>
<BR>
{<BR>
<BR>
/// ;amp;amp;lt;summary;amp;amp;gt;<BR>
<BR>
/// Configuration for caching<BR>
<BR>
/// ;amp;amp;lt;/summary;amp;amp;gt;<BR>
<BR>
public class CachingSection : ConfigurationSection<BR>
<BR>
{<BR>
<BR>
[ConfigurationProperty("CachingTimeSpan", IsRequired = true)]<BR>
<BR>
public TimeSpan CachingTimeSpan <BR>
<BR>
{<BR>
<BR>
get { return (TimeSpan)base["CachingTimeSpan"]; }<BR>
<BR>
set { base["CachingTimeSpan"] = value; }<BR>
<BR>
}<BR>
<BR>
[ConfigurationProperty("FileExtensions", IsDefaultCollection = true, IsRequired = true)]<BR>
<BR>
public FileExtensionCollection FileExtensions <BR>
<BR>
{<BR>
<BR>
get { return ((FileExtensionCollection)base["FileExtensions"]); }<BR>
<BR>
}<BR>
<BR>
}</PRE>
<BR>
为了支持单个的值和集合,我们必须实现一个从<CODE>ConfigurationElementCollection继承的类。</CODE><BR>
<BR>
在我们的例子中,我们需要一个集合来配置可以的扩展列表和他们相应的内容类型。<BR>
<BR>
你可以从这个文件<a href="http://www.software-architects.at/Portals/1/Articles/ASPNETCaching/Code/CachingSection.cs" target="_blank" ><EM><FONT color=#1a8bc8>CachingSection.cs</FONT></EM></A>中下载完整的代码。<BR>
<PRE>  /// ;amp;amp;lt;summary;amp;amp;gt;<BR>
<BR>
/// List of available file extensions<BR>
<BR>
/// ;amp;amp;lt;/summary;amp;amp;gt;<BR>
<BR>
public class FileExtensionCollection : ConfigurationElementCollection<BR>
<BR>
{<BR>
<BR>
...<BR>
<BR>
<BR>
<BR>
}</PRE>
<BR>
最后我们需要一个针对所有扩展的类,一个属性存储扩展,一个属性存储内容类型<BR>
<BR>
<BR>
<PRE>/// ;amp;amp;lt;summary;amp;amp;gt;<BR>
<BR>
/// Configuration for a file extension<BR>
<BR>
/// ;amp;amp;lt;/summary;amp;amp;gt;<BR>
<BR>
public class FileExtension : ConfigurationElement<BR>
<BR>
{<BR>
<BR>
[ConfigurationProperty("Extension", IsRequired = true)]<BR>
<BR>
public string Extension<BR>
<BR>
{<BR>
<BR>
get { return (string)base["Extension"]; }<BR>
<BR>
set { base["Extension"] = value.Replace(".", ""); }<BR>
<BR>
}<BR>
<BR>
[ConfigurationProperty("ContentType", IsRequired = true)]<BR>
<BR>
public string ContentType<BR>
<BR>
{<BR>
<BR>
get { return (string)base["ContentType"]; }<BR>
<BR>
set { base["ContentType"] = value; }<BR>
<BR>
}<BR>
<BR>
}<BR>
<BR>
}</PRE>
<BR>
现在,我们所有要做的就是在webgconfig中添加配置部分。在configSecontions标签中,<BR>
<BR>
我们用<CODE>SoftwareArchitects这个名字</CODE>添加了一个新的<CODE>sectionGroup</CODE>。在这个group中<BR>
<BR>
我们添加了一个名叫caching的section。这个属性的类型能够说明了类和<CODE>CachingSection</CODE> 类<BR>
<BR>
的集合。当然,我们必须在bin文件加中,添加<CODE>CachingSection</CODE> 类的集合。<BR>

<PRE>然后我们可以用这个<BR>
<BR>
配置的标签添加一个新的标签。在这个分组里边,我们用这个section的名字添加了一个新的标签,<BR>
<BR>
并且在这个section中我们在cahingSection类中定义的所有属性现在都可以用了。<BR>
<PRE><configuration><BR>
<BR>
<configSections><BR>
<BR>
<sectionGroup name="SoftwareArchitects"><BR>
<BR>
<section name="Caching" requirePermission="false" type="SoftwareArchitects.Web.Configuration.CachingSection, <BR>
<BR>
SoftwareArchitects.Web.CachingHandler" /><BR>
<BR>
</sectionGroup><BR>
<BR>
</configSections><BR>
<BR>
<SoftwareArchitects><BR>
<BR>
<Caching CachingTimeSpan="1"><BR>
<BR>
<FileExtensions><BR>
<BR>
<add Extension="gif" ContentType="image"gif" /><BR>
<BR>
<add Extension="jpg" ContentType="image"jpeg" /><BR>
<BR>
<add Extension="png" ContentType="image"png" /><BR>
<BR>
</FileExtensions><BR>
<BR>
</Caching><BR>
<BR>
</SoftwareArchitects><BR>
<BR>
...</PRE>

  现在在我们使用 CahingHandler之前,有最后一件需要做的事情。我们必须把它添加到webconfig<BR>
<BR>
中的httpHandlers部分。在那里,我们必须为每个我们想要标记的文件的扩展添加条目到httphandler中。<BR>
<BR>
我决定支持.gif,.jpg,.png的图像。所以我增加了一个为路径*.gif.ashx,*.jpg.ashx 和*.png.ashx<BR>
<BR>
的处理机制。在这个类型中,我指定了Http handler的类和集合。当然这个集合也必须 放在bin文件夹中。<BR>
<PRE> <httpHandlers><BR>
<BR>
<add verb="*" path="*.gif.ashx" type="SoftwareArchitects.Web.CachingHandler, SoftwareArchitects.Web.CachingHandler"/><BR>
<BR>
<add verb="*" path="*.jpg.ashx" type="SoftwareArchitects.Web.CachingHandler, SoftwareArchitects.Web.CachingHandler"/><BR>
<BR>
<add verb="*" path="*.png.ashx" type="SoftwareArchitects.Web.CachingHandler, SoftwareArchitects.Web.CachingHandler"/><BR>
<BR>
</httpHandlers><BR>
<BR>
</configuration></PRE>
<BR>
有也可以使用其他的文件扩展名如*.gifx。但是要这样做,你必须有权限访问IIS去配置新的扩展通过<BR>
<BR>
 <EM>aspnet_isapi.dll来出来。因为空间商没有分配给我配置IIS的权限,所以我必须使用*.ashx,<BR>
<BR>
因为它已经标注在aspnet_isapi.dll中了。<BR>
<BR>
最后我在我的网站中给所有图片添加了扩展名“.ashx”(在css文件和ASPX文件中)。当我再次监视</EM><BR>
<BR>
<a href="http://www.software-architects.com/" target="_blank" ><FONT color=#1a8bc8>http://www.software-architects.com</FONT></A> 的主页,第一次请求仍然产生二十个对服务器的请求,<BR>
<BR>
但是第二次的时候,它只产生了七次请求来载入页面,因为这些图像已经缓存到客户端了。<BR>
<BR>
<BR>
<BR>
<IMG src="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/MNM_withCaching.png"><BR>
<BR>
<BR>
<BR>
你可以从我的网站的这个网址 <a href="http://www.software-architects.at/TechnicalArticles/CachinginASPNET/tabid/75/language/de-AT/Default.aspx" target="_blank" ><FONT color=#1a8bc8>http://www.software-architects.at/TechnicalArticles/CachinginASPNET/tabid/75/language/de-AT/Default.aspx</FONT></A>. <BR>
<BR>
来查看他们怎么工作的。在一个图片上右击并且打开属性对话框。你将看到,那个URL是以.ashx结尾的。<BR>
<BR>
当你右击并且选择“将图片另存为(Save Picture as)”,刚才那个文件名不包括.ashx的扩展,这是因为 content-dispositiont头<BR>
<BR>
当然,你也可以用这个处理机制来处理其他的文件例如 css和javascript文件。因此你能又一次减少请求的数量。<BR>
<BR>
<PRE><STRONG>测试 CachingHandler</STRONG></PRE>
<BR>
  你可以很轻松的用简单的站点来测试缓存图像。我在vs解决方案中用<CODE>CachingWebSite的名字添加了一个网站工程</CODE>,你可以试试他怎么工作的<BR>
<BR>
(下载完整的解决方案<a href="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/CachingHandler.zip" target="_blank" ><FONT color=#1a8bc8>download complete solution</FONT></A>)一方面,这个站点包含一defalut.aspx页面,页面里有个image标签。你可以看到<BR>
<BR>
那个图片是以.ashx结尾的。<BR>
<DIV><img src="/Portals/1/App_Themes/Standard/Images/LogoWithMenuBackground.png.ashx" /><BR>
<BR>
另外一方面,这个网站包含了一个主题Stanard和一个样式表文件。在这个样式表中我使用了一个背景图像。同样这个图片也是以.ashx结尾。<BR>
<BR>
<PRE>body {margin: 0px;padding: 0px;background-image: url(Images/MenuBackground.png.ashx);background-repeat: repeat-x;}<BR>
<BR>
在web.congfig中,我插入了自定义的部分配置<CODE>CachingHandler </CODE>并且一个扩展一个http handler,<BR>
<BR>
和上边解释的一样。此外我在System.web部分添加了一个跟踪(trace)标签来跟踪对每个文件的每个请求<BR>
<DIV><trace enabled="true" pageOutput="false" requestLimit="50" mostRecent="true" />当我运行网站,我看到带着logo的defaul.aspx页面,还有定义在css中的背景图像。<BR>
<BR>
</DIV>

</PRE>
<BR>
</DIV>

<IMG src="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/CachingWebSite.png"><BR>

为了看到跟踪信息,我在ie中打开一个新的标签并且在url中用Trace.axd替换default.aspx这个信息显示了打开default.aspx需要有四个必要的请求。第一次和用户每次按F5的时候,所以文件被发送到客户端。<BR>
<IMG src="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/Trace_withoutCaching.png"><BR>
<BR>
当我转回到第一个标签,有三种可能重新载入页面,我可以<BR>

<BR>
<UL>
    <LI>按F5<BR>
    
    <LI>单击 "Reload page ..." 链接
    <LI>单击 "Reload page ..." 按钮
</LI></UL>

按f5可以重新载入所有的内容,但是当点击链接或者按钮的时候仅仅是重新载入那些没有被缓存<BR>

到客户端的内容。在下边的截图中,我点击了链接和按钮。正如你看到的,只有对default.aspx<BR>

<IMG src="http://www.codeproject.com/KB/aspnet/CachingImagesInASPNET/Trace_withCaching.png"><BR>


如果用户通过超链接或者是后退进入这个页面,那么只有没有被缓存到客户端的文件才会从服务器上请求。<BR>


第5个和第6个请求是通过点击链接引起的,而第7个和第8个是单击按钮引起的。</PRE>
<BR>
</PRE>
喜欢0 评分0
A friend is never known till a man has need. ...CL
游客

返回顶部