把aspx编译成dll文件

转载 2007年10月12日 13:04:00

Windows下窗口编程的界面,为开发大型网络应用程序功能提供了良好的编程接口,也能够极大地提高开发人员的工作效率。

 

然而,一次转换,两次编译过程使得aspx文件在首次执行(或更新后首次运行)时显得略有不足,特别是在拥有大量aspxcodebehind的代码文件的应用环境中,把aspx文件编译成DLL(在.Net中,被称为应用程序集)后再发布,省去一次转换、一次编译的时间及CPU占用率,对提高WEB服务的整体性能会有较大的提升。当然,编译成DLL后,对源代码的保密性也有一定程度的提高。

 

本文通过对Asp.Net的基本处理流程及一个偶然发现的秘密的分析,介绍了在Asp.Net中如何建立aspxDLL的映射,如何开发一个可以处理HTTP请求/响应的DLL,以及如何设置陷阱,把现成的单个aspx文件与codebehindaspx文件编译成DLL的过程,文章最后,还介绍了一个在实际操作过程的小技巧。

 

由于本文要涉及Asp.Net应用程序、命令行编译、web.config配置文件等概念,为了使读者能更好地理解本文内容,也为了使本文看上去不显累赘,先就本文相对应的系统环境作一介绍:

 

系统环境:

Win2000SP3+ IIS5 + .Net Framework 1.0(中文版)。

服务器名称:

由于本文的例子均在本机上测试,服务器名称为localhost

IIS设置:

建立虚拟目录dlltest(真实路径为w:/wwwroot/dlltest),并把它设为应用程序,在dlltest下建立bin目录。所有源文件将放在dlltest目录下,而所有dll文件将放在dlltest/bin目录下。

Asp.Net应用程序配置文件--web.config

dlltest目录下建立一个web.config文件,初始时该文件内容如下:

<?xml version="1.0"?>

<configuration>

<system.web />

</configuration>

命令窗口(DOS窗口)

打开命令窗口,并用cd命令使当前目录为w:/wwwroot/dlltest

一、建立aspxdll的映射

 

首先让我们来看看一般情况下aspx文件是如何被Asp.Net处理的:

当一个HTTP请求(例如“http://webserver/webapp/webpage.aspx”)从客户端发送到IIS服务器时,IIS捕获并分析这个请求,当它分析到这个请求是一个aspx页面时,立即以“/webapp/webpage.aspx”为参数调用Asp.Net运行环境(aspnet_wp.exe,Asp.Net环境启动后,检查“/webapp/webpage.aspx”是否存在,若不存在,则向客户端返回HTTP 404(File not found)错误,否则在Asp.Net 的临时目录中查找相应的dll文件,若不存在或者该dllaspx源文件,则调用csc编译器(若aspx的服务端脚本语言是VBJScript,则调用相应的vbc编译器, jsc编译器)把aspx文件编译成dll,然后Asp.Net再调用该dll来处理具体的客户请求,返回服务器响应。

 

从这个处理流程可以看出,一般情况下,Asp.Net运行环境会自动识别、检查、更新与aspx相对应的dll。那么有没有其它办法可以强制把对一个aspx文件的处理路由到一个已编译存在的DLL呢?方法就是在Asp.Net应用程序配置文件web.configsystem.web节的httpHandlers节添加aspxdll的映射项,语法如下:

<add verb="*" path="aspx文件名" type="类名,dll文件" />

aspx文件:需要被路由的虚拟名称,扩展名必须是aspx,否则IIS会先于Asp.Net运行环境处理该文件。

dll文件: dll文件(应用程序集)的名称,不必输入“.dll”ASP.NET 首先在应用程序的专用 /bin 目录中搜索程序集 DLL,然后在系统程序集缓存中搜索程序集 DLL

类名: 由于一个dll可能会有多个名称空间或多个类,因此必须指明当dll调用时自动加载哪个类。

 

例如,某一Asp.Net应用程序的web.config文件如下:

<?xml version="1.0"?>

<configuration>

<system.web>

<httpHandlers>

<add verb="*" path="index.aspx" type="BBS.IndexPage, bbs" />

</httpHandlers>

</system.web>

</configuration>

 

该配置文件告诉Asp.Net,在客户端请求本应用程序的index.aspx文件时,直接调用应用程序bin目录下的bbs.dll,并自动加载其中的BBS.IndexPage类。

 

二、开发能处理HTML页面的DLL

 

应该指出的是,并不是所有的应用程序集DLL都能实现HTTP请求/响应模式。还是来看一下Microsoft Asp.Net快速入门教程(http://chs.gotdotnet.com/quickstart/aspplus/)中关于“Http 处理程序和工厂的描述:

 

ASP.NET 提供低级别的请求/响应 API,使开发人员能够使用 .NET 框架类为传入的 HTTP 请求提供服务。为此,开发人员需创作支持 System.Web.IHTTPHandler 接口和实现 ProcessRequest() 方法的类。当处理 HTTP 请求不需要由高级别的页框架抽象化提供的服务时,处理程序通常很有用。处理程序的常用用途包括筛选器和类似 CGI 的应用程序,尤其是那些返回二进制数据的应用程序。

 

ASP.NET 收到的每个传入 HTTP 请求最终由实现 IHTTPHandler 的类的特定实例来处理。IHttpHandlerFactory 提供了处理 IHttpHandler 实例 URL 请求的实际解析的结构。除了 ASP.NET 提供的默认 IHttpHandlerFactory 类外,开发人员还可以选择创建和注册工厂以支持大量的请求解析和激活方案。

 

从这段文字可以看出,当aspx页面不涉及.net框架提供的高级界面技术(如数据缓存、状态保持、Web窗体控件引用等等)时,且向客户端输出的不是复杂的HTML文本,特别是只向客户端返回二进制数据(如图片,声音等)时,可以用一个.cs应用程序文件(本文使用c#语言,如果是用VBJScript...)来替代,而该应用程序必须有一个实现System.Web.IHTTPHandler 接口和并实现 ProcessRequest() 方法的类。一个简单的例子如下:

 

/**//* 源文件:ex1.cs 开始 */

 

using System.Web;

 

namespace
 DllTest

...
{

/**/
/*

类必须实现IHttpHandler接口。如果程序将访问会话状态(Session),则必须实现 IRequiresSessionState 接口(不包含任何方法的标记接口)。


*/


public class Ex1Page : IHttpHandler

...
{

/**/
/*

IsReusable
属性告诉.Net框架,本程序是否可以被多个线程同时使用。


true
对应是;false对应否。

*/


public bool IsReusable

...
{

get ...return true
; }



 

/**/
/*

实现ProcessRequest方法,向客户端返回响应数据。


本例中向客户端返回一个简单的HTML页面

*/


public void ProcessRequest(HttpContext context) 

...
{

HttpResponse res = context.Response;

 

res.Write("<html><body>");

res.Write("<h1>DllTest - Ex1
(例1
</h1><hr>");

res.Write("
本页面直接由DLL处理
");

res.Write("</html></body>");

}

 

}

}

 

/**//* 源文件:ex1.cs 结束 */


在命令行状态,用如下的编译命令把ex1.cs编译成ex1.dll,并把它存放在bin目录下。

csc /t:library /out:bin/ex1.dll ex1.cs

 

在配置文件web.config中添加aspx->dll映射,添加后,web.config应该是这样子的:

<?xml version="1.0"?>

<configuration>

<system.web>

<httpHandlers>

<add verb="*" path="dlltest1.aspx" type="DllTest.Ex1Page, ex1" />

</httpHandlers>

</system.web>

</configuration>

 

现在当浏览器访问http://localhost/dlltest/dlltest1.aspx时,实际上就是调用了ex1.dllDllTest.Ex1Page类的ProcessRequest方法,在浏览中应该可以看到一个简单的页面。

 

 

三、把单个aspx文件编译成DLL

 

从上一节微软公开描述的言外之意来看,微软是不支持让开发人员直接把aspx文件编译成DLL的。然而,Asp.Net高级界面技术(服务端HTML控件,WEB控件等等)都是需要通过aspx文件才能展现出来的,如果为了DLL的运行效率而放弃aspx的高级特性,则显然是得不尝失的。

 

现在静下心来分析一下:

csc编译器只是一个c#语言的编译器,它只能对符合C#语言规范的文件进行编译,而aspx文件的格式显然不符合c#语言规范,所以csc编译器是无法对aspx源文件进行编译的。

因此,要想把aspx文件编译成dll文件,必然要先把aspx文件转化成csc编译器能识别的cs源文件。那么用什么工具来进行转换呢?虽然我深信这个工具一定是隐藏在.Net Framework里面,但在查阅了大量的Asp.Net.Net的公开文档及参考手册,资料之后,仍找不到相关资料。

 

呵呵,天无绝人之路,一个偶然的机会,还是让我发现了这个秘密。

 

来看看源文件ex2.aspx:

 

/**//* 源文件:ex2.aspx 开始 */

 

<%@ Page Language="c#" %>

 

<script runat="server">

 

/**//*

你没看错,下一行就是“abcdefg”,正是这一行,才让我有机会写出本篇文章^_^


在文中,我把这一行称作代码陷阱

*/


 

abcdefg 
// 代码陷阱

 

void Page_Load(Object src, EventArgs args)

...
{

if( !IsPostBack ) NoteLabel.Text = "请输入您的姓名:
";

}

 

void
 OnNameSubmit(Object src, EventArgs args)

...
{

string
 name = f_Name.Value;

NoteLabel.Text = (name=="") ? "
姓名不能为空" : name +",您好。欢迎光临!
";

}

</script>

 

<html>

<body>

<form runat="server">

<h1>DllTest - Ex2
(例2
</h1>

<hr>

<asp:label runat="server" id="NoteLabel" style="color:red; font-weight:bold" />

<input runat="server" id="f_Name" size="8">

<button runat="server" onserverclick="OnNameSubmit">
确定
</button>

</form>

</body>

</html>

 

/**//* 源文件:ex2.aspx 结束 */


如果把代码陷阱注释掉或删掉,那么ex2.aspx就是一个简单的Asp.Net文件,用IE浏览此页面可以发现它能正常工作。

 

现在让我们打开陷阱,来看看Asp.Net到底返回了什么?

 

返回的是一个编译错误的页面,报告源文件无法通过编译。让我们感兴趣的是该页面最下方的一个名为显示完整的编译源的超链接,点击些链接,就能看到这个由ex2.aspx转换而来的cs源文件(完整的编译源)的完整内容。把这部分完整的编译源去掉前面的行号信息和其它的一些编译开关(主要是#line编译命令),并关闭那个可爱的代码陷阱(用//把它注释掉或直接把它delete也行),整理后保存为ex2_aspx.cs

 

/**//* 源文件:ex2_aspx.cs 开始 */

 

/**//*

从下面的说明可以看出,确实有一个未公开的工具来完成把aspx文件转化成cs源文件


*/


//------------------------------------------------------------------------------

// <autogenerated>

// This code was generated by a tool.

// Runtime Version:1.0.3705.0

//

// Changes to this file may cause incorrect behavior and will be lost if 

// the code is regenerated.

// </autogenerated>

//------------------------------------------------------------------------------

 

/**//* 

奇怪的是:命名空间居然是ASP而不是
ASPX 

建议把该名称改成适合应用程序的名称,防止命名冲突,例如针对本文,可以改成
DllTest

这里没改是为了让大家看清它的原貌


*/


namespace ASP ...{

using
 System;

using
 System.Collections;

using
 System.Collections.Specialized;

using
 System.Configuration;

using
 System.Text;

using
 System.Text.RegularExpressions;

using
 System.Web;

using
 System.Web.Caching;

using
 System.Web.SessionState;

using
 System.Web.Security;

using
 System.Web.UI;

using
 System.Web.UI.WebControls;

using
 System.Web.UI.HtmlControls;

 

/**/
/*

1
、注意一下类名的构成,如果必要,可以把它改成有意义的名称,例如针对本文,可以改成
Ex2Page

2
、注意它的基类。Syste.Web.UI.Page实现了IHttpHandler接口,由于要访问Session,所以也实现了IRequiresSessionState接口。


*/
 

public class ex2_aspx : System.Web.UI.Page, System.Web.SessionState.IRequiresSessionState ...
{

 

private static int
 __autoHandlers;

protected
 System.Web.UI.WebControls.Label NoteLabel;

protected
 System.Web.UI.HtmlControls.HtmlInputText f_Name;

protected
 System.Web.UI.HtmlControls.HtmlButton __control3;

protected
 System.Web.UI.HtmlControls.HtmlForm __control2;

private static bool __intialized = false
;

private static
 System.Collections.ArrayList __fileDependencies;

 

/**//* 现在可以关掉陷阱 */


// abcdefg 

 

void Page_Load(Object src, EventArgs args)

...
{

if( !IsPostBack ) NoteLabel.Text = "请输入您的姓名:
 ";

}

 

void
 OnNameSubmit(Object src, EventArgs args)

...
{

string
 name = f_Name.Value;

 

NoteLabel.Text = (name=="") ? "
姓名不能为空" : name +",您好。欢迎光临!
";

}

 

/**//* 构造函数 */
 

public ex2_aspx() ...
{

System.Collections.ArrayList dependencies;

if ((ASP.ex2_aspx.__intialized == false)) ...
{

dependencies = 
new
 System.Collections.ArrayList();

 

/**/
/* 

应该把下面这行注释掉,让DLL成为一个无依赖的独立文件


防止在DLL运行时再次去查找、比较它的依赖文件的新旧

*/


//dependencies.Add("W:/wwwroot/dlltest/ex2.aspx");

ASP.ex2_aspx.__fileDependencies = dependencies;

ASP.ex2_aspx.__intialized = 
true;

}

}

 

protected override int AutoHandlers ...
{

get ...
{

return
 ASP.ex2_aspx.__autoHandlers;

}

set ...
{

ASP.ex2_aspx.__autoHandlers = value;

}

}

 

protected System.Web.HttpApplication ApplicationInstance ...
{

get ...
{

return ((System.Web.HttpApplication)(this
.Context.ApplicationInstance));

}

}

 

public override string TemplateSourceDirectory ...
{

get ...
{

return
 "/dlltest";

}

}

 

private System.Web.UI.Control __BuildControlNoteLabel() ...
{

System.Web.UI.WebControls.Label __ctrl;

__ctrl = 
new
 System.Web.UI.WebControls.Label();

this
.NoteLabel = __ctrl;

__ctrl.ID = "NoteLabel";

((System.Web.UI.IAttributeAccessor)(__ctrl)).SetAttribute("style", "color:red; font-weight:bold");

return
 __ctrl;

}

 

private System.Web.UI.Control __BuildControlf_Name() ...
{

System.Web.UI.HtmlControls.HtmlInputText __ctrl;

__ctrl = 
new
 System.Web.UI.HtmlControls.HtmlInputText();

this
.f_Name = __ctrl;

__ctrl.ID = "f_Name";

__ctrl.Size = 8;

return
 __ctrl;

}

 

private System.Web.UI.Control __BuildControl__control3() ...
{

System.Web.UI.HtmlControls.HtmlButton __ctrl;

__ctrl = 
new
 System.Web.UI.HtmlControls.HtmlButton();

this
.__control3= __ctrl;

System.Web.UI.IParserAccessor __parser = ((System.Web.UI.IParserAccessor)(__ctrl));

__parser.AddParsedSubObject(
new System.Web.UI.LiteralControl("确定
"));

__ctrl.ServerClick += 
new System.EventHandler(this
.OnNameSubmit);

return
 __ctrl;

}

 

private System.Web.UI.Control __BuildControl__control2() ...
{

System.Web.UI.HtmlControls.HtmlForm __ctrl;

__ctrl = 
new
 System.Web.UI.HtmlControls.HtmlForm();

this
.__control2= __ctrl;

System.Web.UI.IParserAccessor __parser = ((System.Web.UI.IParserAccessor)(__ctrl));

__parser.AddParsedSubObject(
new System.Web.UI.LiteralControl("  <h1>DllTest - Ex2(例2
</h1>  <hr>  "));

this
.__BuildControlNoteLabel();

__parser.AddParsedSubObject(
this
.NoteLabel);

__parser.AddParsedSubObject(
new
 System.Web.UI.LiteralControl("  "));

this
.__BuildControlf_Name();

__parser.AddParsedSubObject(
this
.f_Name);

__parser.AddParsedSubObject(
new
 System.Web.UI.LiteralControl("  "));

this
.__BuildControl__control3();

__parser.AddParsedSubObject(
this
.__control3);

__parser.AddParsedSubObject(
new
 System.Web.UI.LiteralControl("  "));

return
 __ctrl;

}

 

private void __BuildControlTree(System.Web.UI.Control __ctrl) ...
{

System.Web.UI.IParserAccessor __parser = ((System.Web.UI.IParserAccessor)(__ctrl));

__parser.AddParsedSubObject(
new
 System.Web.UI.LiteralControl(" <html> <body>  "));

this
.__BuildControl__control2();

__parser.AddParsedSubObject(
this
.__control2);

__parser.AddParsedSubObject(
new
 System.Web.UI.LiteralControl(" </body> </html> "));

}

 

protected override void FrameworkInitialize() ...
{

this.__BuildControlTree(this
);

this
.FileDependencies = ASP.ex2_aspx.__fileDependencies;

this.EnableViewStateMac = true
;

}

 

public override int GetTypeHashCode() ...
{

return
 -11574299;

}

}

}

 

/**//* 源文件:ex2_aspx.cs 结束 */


相信大家在分析了这个文件之后,会对Asp.Net运行原理有更进一步的认识(与本文无关,不详述)。

 

在命令行状态,用如下的编译命令把ex2_aspx.cs编译成ex2.dll,并把它存放在bin目录下。

csc /t:library /out:bin/ex2.dll ex2_aspx.cs

 

在配置文件web.config中添加aspx->dll映射,即在system.web节的httpHandlers添加下面一行:

<add verb="*" path="dlltest2.aspx" type="ASP.ex2_aspx, ex2" />

 

 

现在当浏览器访问http://localhost/dlltest/dlltest2.aspx时,就如同访问ex2.aspx一样。当然,现在即使ex2.aspx不存在,或者已经更新过,也不会对页面访问有任何影响,除非重新生成bin/ex2.dll

 

 

四、把codebehindaspx文件编译成dll

 

对于把codebehindaspx文件编译成dll,其中把aspx文件转化成cs源文件的原理同上,也是先设置一个代码陷阱,然后把完整的编译源进行适当整理,保存为cs源文件。区别是在编译成dll时的步骤:(为叙述方便,假设界面文件为ex3.aspxcodebehind文件为ex3.aspx.csex3.aspx完整编译源保存为ex3_aspx.cs

第一步:先用如下命令把ex3.aspx.cs编译成bin/ex3.aspx.cs.dll

csc /t:library /out:bin/ex3.aspx.cs.dll ex3.aspx.cs

第二步:再用如下命令把ex3_aspx.cs编译成bin/ex3.dll

csc /t:library /r:bin/ex3.aspx.cs.dll /out:bin/ex3.dll ex3_aspx.cs

 

然后在配置文件web.config中添加aspx->dll映射,即在system.web节的httpHandlers添加下面一行:

<add verb="*" path="dlltest3.aspx" type="ASP.ex3_aspx, ex3" />

 

现在打开浏览器,访问http://localhost/dlltest/dlltest3.aspx试试。

 

五、一点小技巧

 

在设置陷阱aspx文件转化成cs源文件时,一般是使用copypaste方法把完整的编译源保存在记事本或vs.net或其它asp.net开发环境,再进行整理后保存为cs源文件的。

整理,就是把paste进来的行号信息与“#line”编译指令去掉。如果是手动地删掉这些信息,则会太麻烦,即使是一个简单的如ex2.aspx的文件,也会产生约270行的完整的编译源

我所使用的一个小技巧是:在记事本里,用替换的方法来快速整理。用"/* "来全部替换"",用"*/"来全部替换"",用"// #line "来全部替换"#line",替换完成之后,再把代码陷阱注释掉,把主类构造函数里设置依赖文件的语句全部注释掉,这样就算整理完成了。

 

 

如何编译生成dll文件以及如何调用dll文件

点击确定选择dll选项和空项目选项 点击完成  1.2 为项目添加c++源文件mydll.cpp extern "C" // 此处extern "c" 为解决c/c++兼容问题 { _...
  • little_bobo
  • little_bobo
  • 2017年02月14日 19:52
  • 1439

由.o文件生成.a文件(静态库)或者.dll文件(动态库)

关于库生成的问题     我们通常把一些公用函数制作成函数库,供其它程序使用。函数库分为静态库和动态库两种。静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库。动态库在程序编译时并...
  • zhuzeji
  • zhuzeji
  • 2014年11月22日 17:03
  • 344

Go1.10在编译成dll小实例

package mainimport ( "net" ) //必须导入 import "C"//编译成动态库也是必须的 func main() {}//export Interfaces fu...
  • fyxichen
  • fyxichen
  • 2017年12月08日 18:56
  • 257

asp.net 预编译后代码的反编译成源码(演示:鹏为CRM E4版本)

.net 破解新人贴。其实这也不算什么破解,但是对新手来说还是有一定难度滴。。。 很多人会被这样的情况吓到。。   打开后只有一句中文,其他的代码都没有。但是其中有一个很重要的信息,它告诉了我们...
  • u011047160
  • u011047160
  • 2015年06月29日 16:02
  • 1996

MVC项目开发中那些用到的知识点(将cshtml文件编译成dll文件)

在做Asp.Net  Mvc项目的时候,发现如果两个完全独立的项目如果想共享调用cshtml,不知道如何处理了。于是就上网百度、Google了一下,结果答案就出来了。 首先需要下载一个工具工具链接 ...
  • dyllove98
  • dyllove98
  • 2013年04月04日 00:05
  • 4308

matlab与C++混合编程 matlab发布、打包DLL

本文主要讲述将matlab编写好的m函数导出成dll,以供C++程序调用。使用matlab命令手动打包和matlab工具箱中deploytool的使用。写本文的主要原因是,在项目过程中遇到一个棘手问题...
  • damant
  • damant
  • 2016年03月28日 22:38
  • 2374

使用WinRar将Qt编译生成的exe和依赖的dll文件打包为一个exe文件的简单方法

如题,需要将QtCreator编译生成的*.exe文件和它所依赖的动态链接库(*.dll)文件打包为一个exe文件,步骤如下: (1)将编译生成的AppTest.exe、依赖的库文件、图标文件等复制...
  • zgrjkflmkyc
  • zgrjkflmkyc
  • 2015年11月10日 10:57
  • 2098

如何编译生成dll

动态链接库是Windows的基石。所有的Win32 API函数都包含在DLL中。3个最重要的DLL是KERNEL32.DLL,它由管理内存、进程和线程的函数组成;USER32.DLL,它由执行用户界面...
  • qianchenglenger
  • qianchenglenger
  • 2014年03月25日 18:52
  • 49782

编辑IL文件 修改DLL文件

本文章只是技术探讨,学习,技术上的研究而已。请支持正版. 如:KS.Gantt.DLL 为例 使用ILSpy反编译 工具利用ildasm反编译 KS.Gantt.dll  生成IL中间...
  • rztyfx
  • rztyfx
  • 2016年11月23日 16:22
  • 431

jar 转换成dll

将jar转换为dll 以前总是感觉 是不可以运用的 但是 后来经过上网上搜一些资料后然后可以运用了。 具体运用如下步骤 : 主要运用 需要    下载一个 IKVM   配置好环境变量  具...
  • a351945755
  • a351945755
  • 2014年03月24日 14:06
  • 6869
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:把aspx编译成dll文件
举报原因:
原因补充:

(最多只允许输入30个字)