从 .NET 程序集提供图像(一)

程序

摘要:大多数 Web 站点的用户界面都需要有图像,这些图像通常存储在磁盘上。本文介绍如何从程序集提供图像。从程序集提供图像可以避免众多文件散布在磁盘上,简化 Web 服务器的安装和配置,以及提高图像的安全性。(本文包含一些指向英文站点的链接)

适用于:
Microsoft .NET Framework 版本 1.0 和 1.1
Microsoft Visual C#
Microsoft ASP.NET

下载 MFRImages.exe 示例文件。下载内容中包括 readme.htm,用于说明如何配置示例。

目录:
简介
问题概述
提供图像
从 ASPX 页面提供图像
从自定义处理程序提供图像

简介

经常会听到那些编写过 Windows 控件,后来又转向 Web 的程序员抱怨:“为什么不能像控件那样将图像存储在同一个程序集内呢?” 答案是可以,您只需要知道如何做到这一点。本文介绍如何从一个程序集内提供图像,并提供了两种检索图像的方法。下载本页顶端的 MFRImages.exe,以便完整地了解下面讨论的示例代码。

问题概述

Web 站点上的图像通常通过 URL 来引用,例如 。/uploadpic/2007-2/2007262311518.gif它告诉 Web 客户端到哪里去查找图像。在 Web 页中,图像与文字是分别下载的。图像通常存储在 Web 服务器中命名为 /images 等名称的子目录中,页面只是提供到这些图像的引用,以使它们能够显示在客户端的浏览器中。

当创建自定义控件时,情况也是如此 - 图像被用户下载之前通常需要存储在磁盘上。作为控件的编写者,您可能希望能够提供一个程序集,其中不仅包括控件,而且还包括相应的默认图像。控件保持独立完整很有益,因为它只需要用户进行较少的配置。(负责配置的用户可能会忘记将大量图像文件复制到生产服务器上,从而导致其他用户的不满。)

希望将资源绑定到程序集的另外一个原因可能是为了确保用户不能更改这些资源 - 例如对于公司商标,应始终使用特定的图像,而不能由某个用户将错误的图像(如不正确的字样或颜色)误保存到磁盘中。如果已对您的程序集进行了严格命名,当加载该程序集时,可以进行测试以确保该程序集未被篡改。

从程序集提供图像的主要问题是 HTTP 需要通过 URL 获得图像 - 您不能只将直接插入 HTML 的一串字节返回给用户,就期望能够正确显示图像。设法将图像请求重定向到程序集内的某个资源是很有必要的,在本文中我将介绍两种方法。

在继续下面的内容之前,还要注意一个问题。如果在 Web 服务器上没有进行其他一些配置,想提供一个能够正确呈现图像的完全独立控件是不可能的。您还需要在服务器上创建其他的文件(至少一个),或对 IIS 配置数据库做一些更改,以将图像提供给客户端。不过,完成这些简单的更改后,您就可以轻松地从任何程序集提供图像了。

提供图像

既然问题的关键是要提供图像,那么我们现在就进入正题。当从调用方传来信息后,我们需要找到一种方法来加载图像并在响应流中返回图像。请注意,此示例中提供的图像只有 .gif 一种类型 - 要提供其他类型的图像,则需要通过图像扩展名来推断内容类型和图像格式。

以下函数显示如何从给定的程序集加载图像,并通过 HttpResponse 流返回给客户端。我们将把此函数作为后面代码的基础,并在接下来的内容中添加更有用的功能,例如异常处理和图像缓存。我将把此函数定义为 ManifestImageLoader 类中的静态函数。

public class ManifestImageLoader
{
public static void RenderImage (
string assembly , string image , HttpContext context )
{
Assembly resourceAssem = Assembly.Load ( assembly ) ;

// 获取资源
using ( Stream imageStream =
resourceAssem.GetManifestResourceStream ( image ) )
{
// 如果可以,将其写出
using ( System.Drawing.Image theImage =
System.Drawing.Image.FromStream ( imageStream ) )
response.ContentType = "image/gif" ;
theImage.Save ( context.Response.OutputStream , ImageFormat.Gif ) ;
}
}
}

函数 RenderImage 接受程序集名称、图像名称和响应流。有了有效的程序集名称和图像名称,加载图像并将其返回到输出流就轻而易举了。首先加载程序集,然后使用 Assembly.GetManifestResourceStream 函数返回已命名资源(在此实例中为图像)的数据流。您需要为 System.Reflection 和 System.IO 添加 using 子句,使其能够通过编译,并引用 System.Drawing 程序集。

有了图像数据流之后,可以使用 Image.FromStream() 方法从该字节流构造图像 - 请注意,我们使用的是 System.Drawing 中的图像类,而不是 System.Web.UI.WebControls 中具有类似名称的类,因为前者具有访问 Win32 图像函数的权限,而后者在 Web 控件中包含了 标记。

您或许不熟悉 C# using 语法,该语法在 Try/Finally 块中包含了代码,可以确保对括号中的项调用 Dispose。现在我们可以从程序集提供图像了,首先需要能够为该图像创建 URL。第一种方法是使用 .ASPX 页面。

从 ASPX 页面提供图像

第一种提供图像的方法需要使用 .ASPX 页面,该页面通常要驻留在服务器上的某个位置。页面本身不包含内容 - 它的主要功能是从 URL 检索参数,并使用这些参数来检索图像。

例如,有一个名为 ImageFromASPX.aspx 的页面位于 Web 站点的根目录下。然后可以在 HTML 中使用以下 URL 编码语法来定义从此页面提供的所有图像。这里我们提供的是名为 winxp.gif 的标题图像。

<img src=http://www.163design.net/n/a/"/imagefromASPX.aspx?assem=ImageServerℑ=winxp.gif" />

在 URL 中,我们定义了 ASPX 页面的路径,然后定义了两个参数,一个是程序集名称(在此实例中为 ImageServer),另一个是图像名称。这在安全性方面可能存在风险,因此在本文的后面我将介绍一种可用于此数据的加密方法。

在 ASPX 页面的代码中,可以写入以下内容:

private void Page_Load ( object sender, System.EventArgs e )
{
// 检索参数
string assembly = Request.QueryString["assem"] ;
string image = Request.QueryString["image"] ;

// 并加载图像
ManifestImageLoader.RenderImage ( assembly , image ) ;
}

代码所做的就是根据请求分析参数,然后调用我们前面编写的 RenderImage 函数。正如您看到的,这并不难实现。但它有一个缺点,即所有对图像的请求都需要通过同一个 URL。也就是说,每个自定义控件都必须知道 imageserver.aspx 文件的位置和名称,才能提供图像。如何避免这个限制是下一节的主题。

从自定义处理程序提供图像

如果您以前从未接触过处理程序,我将在这里大概介绍一下。处理程序是用于实现 IHttpHandler 接口的对象。当包含给定文件扩展名的请求通过 ASP.NET 管道时,将为特定动词(如 POST、GET 等)或一组动词调用处理程序。

通常,ASP.NET 会检查请求的文件扩展名,并将请求传送给与该扩展名相关联的处理程序。了解了这些知识,我们就可以创建处理程序,将其与自己的文件扩展名相关联(这样,ASP.NET 就会知道我们要调用的是处理程序,而不是其他内容),并且以该方式提供图像。

下面的代码显示的是一个简单的处理程序,该处理程序使用了上面声明的 RenderImage 函数。

public class ManifestResourceHandler : IHttpHandler
{
/// <summary>
/// 处理图像请求
/// </summary>
/// <param name="context">The current HTTP context</param>
void IHttpHandler.ProcessRequest ( System.Web.HttpContext context )
{
// 从请求中获取程序集名称和资源名称
string assembly = context.Request.QueryString["assem"] ;
string image = context.Request .QueryString["image"] ;

// 然后加载图像并返回给调用方
ManifestImageLoader.RenderImage ( assembly , image ) ;
}

/// <summary>
/// 此处理程序可以重复使用,不需要循环
/// </summary>
bool IHttpHandler.IsReusable
{
get { return true; }
}
}

这段代码与上面为 ASPX 页面显示的代码非常类似 - 从传来的 URL 读取参数,然后将这些参数传递到 RenderImage 函数。

现在,要想使用处理程序来提供图像,我们需要使用不同的 URL。在此实例中,需要创建一个虚构的文件扩展名(即在 IIS 中不存在的扩展名),这样,图像请求就可以传送给正确的处理程序。在此示例中我将使用扩展名“mfr”(表示“清单资源”)。图像请求现在看起来有点像下面的描述。

<img src=http://www.163design.net/n/a/".mfr?assem=MS.Resourcesℑ=winxp.gif" />

注意,我还未指定资源的路径,只是指定了文件扩展名 .mfr。

使用处理程序的主要好处是可以为所有请求调用该程序,而不用考虑它们的路径。要使处理程序能够工作还需要另外两个步骤。首先,需要修改 web.config,以指定新的处理程序:

<configuration>
<system.web>
...
<httpHandlers>
<add verb="GET" path="*.mfr" type="ImageServer.ManifestResourceHandler, ImageServer" />
</httpHandlers>
</system.web>
</configuration>

上述配置文件中的类型定义了实现处理程序的类型和程序集。注意,动词属性区分大小写,因此应设置为 GET 而不是其他的大小写形式。程序集本身需要驻留在您 Web 站点的二进制目录中,或安装在全局程序集缓存 (GAC) 中。

其次,您需要在 IIS 管理中编辑 Web 服务器的配置。单击您要更改的 Web 站点的 Properties(属性),选择 Home Directory(主目录)选项卡,然后单击 Configuration(配置)。将显示与以下窗口类似的窗口。

单击 Add(添加)按钮,为 .mfr 文件类型创建条目。每个扩展名都会被映射到处理资源请求的 ISAPI 过滤器。对于 ASP.NET,为 aspnet_isapi.dll 过滤器。此库驻留在磁盘中已安装的 Framework 下,因此要设置 .mfr 扩展名的所有请求以通过相应的 ISAPI dll,需要进行如下设置:

版本 路径
1.0 C:\WINDOWS\Microsoft.NET\Framework\v1.0.3705\aspnet_isapi.dll
1.1 C:\WINDOWS\Microsoft.NET\Framework\v1.1.4322\aspnet_isapi.dll

其余的设置显示在以下图像中。请注意,必须清除 Verify that file exists(检查文件是否存在)复选框,否则永远不会调用处理程序(因为磁盘中不存在实际的 .mfr 文件)。

现在应该可以运行处理程序了。在浏览器中键入映射到程序集中的资源的 URL:

如果您接收到的不是请求的图像,而是一个异常(如“‘null’不是‘stream’的有效值”),那么您可能遇到了目前我们还没有在代码中进行处理的一些小问题 - 如果图像有错误怎么办? 我们将在下一节对这个问题及其他一些小问题进行纠正。

时间: 2024-07-31 23:11:20

从 .NET 程序集提供图像(一)的相关文章

从 .NET 程序集提供图像(二)

程序 目录:增强代码整理安全性小结关于作者 增强代码 代码中首先要处理的是大小写形式.HTTP 认为以下所有 URL 都相同,因为 URL 不区分大小写. <img src=http://www.163design.net/n/a/".mfr?assem=ImageServerℑ=winxp.gif" /><img src=http://www.163design.net/n/a/".mfr?assem=ImageServerℑ=winxp.gif"

用SQL Server为Web浏览器提供图像1

微软SQL Server数据库服务器能够在SQL数据库里保存图片和大量的文本.图片和文本使用的数据类型分别是image图片类型和text文本类型.假如使用VB或VC开发前端应用程序,在windows操作系统上运行,那么可以容易地从SQL数据库里提取图片数据,把它放入一个文件,并在屏幕上显示出来.但是,要是想从SQL Server得到图片,并在Web浏览器里显示,就不那么容易了.虽然在微软SQL Server 6.5里提供了Web助理和存储过程 sp_makewebtask.sp_runwebta

用SQL Server为Web浏览器提供图像2

建立ActiveX 动态链接库     使用 VB 5.0,我们要开发一个ActiveX 动态链接库.这个动态链接库让我们能够连接到SQL数据库,提取图片数据,并把它们保存成文件,放在目录里.要建立动态链接库,请在打开VB程序之后执行下列步骤:首先:在文件(File)菜单里,选择新建项目(New Project).接着,在新建立项目对话框里,双击ActiveX DLL 图标.VB自动地向项目里增加一个类,叫做Class1.然后,在属性容器里,双击名称(Name)属性,把它改成clsImageLo

用SQL Server为Web浏览器提供图像

微软SQL Server数据库服务器能够在SQL数据库里保存图片和大量的文本.图片和文本使用的数据类型分别是image图片类型和text文本类型.假如使用VB或VC开发前端应用程序,在windows操作系统上运行,那么可以容易地从SQL数据库里提取图片数据,把它放入一个文件,并在屏幕上显示出来.但是,要是想从SQL Server得到图片,并在Web浏览器里显示,就不那么容易了.虽然在微软SQL Server 6.5里提供了Web助理和存储过程 sp_makewebtask.sp_runwebta

用SQL Server为Web浏览器提供图像3(end)

建立Web项目和设置数据库连接 在注册了 ImageLoader 动态链接库之后,我们就可以建立Web项目了.我们使用微软Visual InterDev 1.0 建立ASP页面以及到微软SQL Server数据库的数据连接. 首先,打开微软Visual InterDev.在文件File 菜单里,选择新建(New)菜单项.在出现的对话框里,选中Web项目向导,在项目名称字段里输入ImageLoad .单击确定(OK),完成后面两步,结束向导. 可以看到, Visual InterDev 在工作空间

Effective C#原则32:选择小而内聚的程序集

这一原则实际应该取这个名字:"应该创建大小合理而且包含少量公共 类型的程序集".但这太沉长了,所以就以我认为最常见的错误来命名: 开发人员总是把所有的东西,除了厨房里水沟以外(译注:夸张说法,kitchen sink可能是个口语词,没能查到是什么意思,所以就直译了.),都放到一个程 序集.这不利于重用其中的组件,也不利于系统中小部份的更新.很多以二进制 组件形式存在的小程序集可以让这些都变得简单. 然而这个标题对于程 序集的内聚来说也很醒目的.程序集的内聚性是指概念单元到单个组件的职责

图像的色彩与色调的调整

一.图像的色彩模式 1.位图模式 位图模式这种形式通常被称为黑白艺术或一位元艺术,它只由黑白两色构成而且没有灰色阴影的图像,按这种方式形成的图像处理速度快,产生的图像文件小,易于操作,因为它所保存图像的颜色信息少.要将图像转换为位图模式,必须首先将图像转换为灰度模式,然后再由灰度模式转换为位图模式. 2.灰度模式 灰度图像中只有灰度颜色而没有彩色,在Photoshop中灰度图可以看成是只有一种颜色通道的数字图像,它可以设置灰阶的级别,如常用的8位/通道,16 位/通道等,其位数的大小代表了通道中

aspnet-未能加载文件或程序集“SharpSvn”或它的某一个依赖项。流入了不在清单中的模块

问题描述 未能加载文件或程序集"SharpSvn"或它的某一个依赖项.流入了不在清单中的模块 说明: 执行当前 Web 请求期间,出现未经处理的异常.请检查堆栈跟踪信息,以了解有关该错误以及代码中导致错误的出处的详细信息. 异常详细信息: System.IO.FileLoadException: 未能加载文件或程序集"SharpSvn"或它的某一个依赖项.流入了不在清单中的模块. (异常来自 HRESULT:0x80131043) 源错误: 执行当前 Web 请求期

“前.NET Core时代”如何实现跨平台代码重用 ——程序集重用

除了在源代码层面实现共享("前.NET Core时代"如何实现跨平台代码重用 --源文件重用)之外,我们还可以跨平台共享同一个程序集,这种独立于具体平台的"中性"程序集通过创建一种名为"可移植类库(PCL: Portable Class Library)"项目来实现.为了让读者朋友们对PCL的实现机制具有充分的认识,我们先来讨论一个被我称为"程序集动态绑定"的话题. 目录 一.何谓程序集动态绑定? 二.程序集一致性 三.程序集