实现自己的ASP.NET宿主系统

.Net 专栏收录该内容
50 篇文章 0 订阅

实现自己的ASP.NET宿主系统

杨山河

 一、 宿主概念
       托管是.NET的一个很基础的概念,所有的.NET应用程序代码要完全发挥作用需要进入托管的环境(CLR --Common Language Runtime),而这个环境实际上就是称作宿主(Host)为将要启动的.NET代码准备的。目前来讲windows系统上,能够担负这个重任的有3类已存程序:
      1、 shell(通常是Explorer),提供从用户桌面启动.NET程序,创建一个进程,启动此进程建立CLR
     2、 浏览器宿主(Internet Explorer),处理从web下载的.NET代码执行。
     3、 服务器宿主(如IIS的辅助进程ASPnet_wp.exe)
     通常来讲,我们开发的ASP.NET的程序运行在IIS的环境下(实际上由一个ISAPI控制启动CLR),但实际上ASP.NET程序可以摆脱IIS单独在任何托管环境下运行。本文讨论了ASP.NET程序如何在自定义的环境中启动,希望有助于我们了解ASP.NET的执行原理,同时使我们开发的ASP.NET能够在任何.NET环境下执行,不管是服务器操作系统还是普通的桌面操作系统。

二、 IIS宿主中ASP.NET的执行分析
      关于IIS中ASP.NET的执行细节,很多文章做了详尽权威的分析,本文不打算赘述,在此给出一些参考:
http://www.yesky.com/SoftChannel/72342380468043776/20030924/1731387.shtml
http://chs.gotdotnet.com/quickstart/ASPplus/doc/procmodel.ASPx
      这些文章大致重点分析了:宿主环将如何启动、ASP.NET应用程序如何产生程序集、如何加载,同宿主的交互等细节。

三、 构造自己的ASP.NET宿主程序
      ASP.NET是作为微软ASP的替代技术出现的,所以我们重点讨论如何通过web方式应用ASP.NET(显然还有其他方式),具体就是:我们用.NET平台的语言编写一个控制台程序,这个程序启动一个ASP.NET应用环境,执行关于ASPx的请求。具体来讲,需要做以下工作:
      1、实现一个Web Server,监听所有的web请求,实现Http web hosting
      2、启动一个应用程序域,创建一个ASP.NET的ApplicationHost,建立一个ASP.NET的应用程序域,另外还建立一个HttpWorkerRequest的具体实现类,该类可以处理ASPx请求,编译ASPx页,编译后的托管代码缓存入当前应用程序域,然后执行代码,得到执行结果。建议在继续阅读下文前,仔细翻查MSDN中的关于这两个类得参考说明。
      System.Web.Hosting.ApplicationHost类用于建立一个独立的应用程序域。当然不是普通的应用程序域,而是为ASP.NET建立执行环境,准备需要的空间、数据结构等。仅有一个静态方法static object CreateApplicationHost(
   Type host   //具体的用户实现类,就是ASP.NET应用域需要加载的类
   string virtualDir,  //此应用域在整个web中的执行目录,虚拟目录
   string physicalDir //对应的物理目录
        );
      而其中的host 参数指向一个具体的类,由于该类实际上属于两个应用域之间的联系类,在两个应用程序域之间编组传递数据,所以必须要继承自MarshalByRefObject,以允许在支持应用程序中跨应用程序域边界访问(至于为什么,建议翻查参考3)。
      可以看到,我们需要启动两个应用程序域(web server功能应用程序域和ASP.NET 应用程序域),而这两个(应用程序)域之间通过跨(应用程序)域的流对象引用来实现,使得在ASP.NET域中执行的结果可以通过web server域返回给请求者。
可以大致下图表达
      执行ASP.NET的Web服务器端

 

 

 


           WEB客户端

代码实现分析:
using System;
using System.Web ;
using System.Web.Hosting;
using System.IO;
using System.NET;
using System.NET.Sockets ;
using System.Text ;
using System.Threading ;

namespace MyIIS
{
 class ASPHostServer
 {  
  [STAThread]
  static void Main(string[] args)
  {
//创建并启动服务器
   MyServer myserver=new MyServer(“/”, ”c://inetpub//wwwroot//myWeb”);
  }
 }

 class MyServer    //处理HTTP协议的服务器类
 {
  private ASPDOTNETHost ASPnetHost; //ASP.NET host的实例
  private TcpListener mytcp;    //Web监听套接字
  bool bSvcRunning=true;    //服务是否运行指示
  FileStream fs;      //处理http请求的普通文本要求

  public MyServer(string virtualDir ,vstring realPath)     
  {//在构造函数中启动web监听服务
   try
   {
    mytcp=new TcpListener(8001);
    mytcp.Start();   //启动在8001端口的监听
    Console.WriteLine("服务启动...");
 //利用CreateApplicationHost方法建立一个独立的应用程序域执行ASP.NET程序
    ASPnetHost = ( ASPDOTNETHost )ApplicationHost.CreateApplicationHost
     ( typeof( ASPDOTNETHost ) , virtualDir , realPath);   
    Thread t=new  Thread(new ThreadStart(MainSvcThread));
    t.Start();  //服务线程启动 负责处理每一个客户端的请求
   }
   catch(NullReferenceException)
   {
    Console.WriteLine("NullReferenceException throwed!") ;
   }
  }

  public void MainSvcThread()  //ASP.NET Host的web服务器的主要服务线程
  {
   int s=0;
   string strRequest;  //请求信息
   string strDir;   //请求的目录
   string strRequestFile; //请求的文件名
   string strErr="";  //错误信息
   string strRealDir;  //实际目录
string strWebRoot=rpath; //应用根目录
   string strRealFile="";  //正在请求的文件的磁盘路径
   string strResponse="";  //回应响应缓冲区
   string strMsg="";   //格式化响应信息
   byte[] bs;     //输出字节缓冲区
   
   while(bSvcRunning)
   {
    Socket sck=mytcp.AcceptSocket(); //每个请求到来
    if(sck.Connected)
    {
     Console.WriteLine("Client {0} connected!",sck.RemoteEndPoint);
     byte[] bRecv=new byte[1024]; //缓冲区
     int l=sck.Receive(bRecv,bRecv.Length,0);                   
     string strBuf=Encoding.Default.GetString(bRecv);  //转换成字符串,便于分析
     s=strBuf.IndexOf("HTTP",1);
     string httpver=strBuf.Substring(s,8); // HTTP/1.1 之类的
     strRequest=strBuf.Substring(0,s-1); 
     strRequest.Replace("//","/");
     if((strRequest.IndexOf(".")<1) && (!strRequest.EndsWith("/")))
     {
      strRequest += "/";
     }
     s=strRequest.LastIndexOf("/")+1;
     strRequestFile = strRequest.Substring(s);     strDir=strRequest.Substring(strRequest.IndexOf("/"),strRequest.LastIndexOf("/")-3);  //取得访问的URL                   
     if(strDir=="/")
     {
      strRealDir=strWebRoot;
     }
     else
     {
      strDir=strDir.Replace("/","//");
      strRealDir=strWebRoot + strDir;
     }
     Console.WriteLine("Client request dir: {0}" , strRealDir);  
     if(strRequestFile.Length==0)
     {
      strRequestFile="default.htm";   //缺省文档
     }
     int iTotlaBytes=0;   //总计需要输出的字节
     strResponse="";   //输出内容
     strRealFile = strRealDir +"//"+ strRequestFile;
     if(strRealFile.EndsWith(".ASPx"))  //这里有Bug!!
     {
      string output="";
//注意我下面的语句们给host对象ProcessRequest方法传递了一个ref类型的参数,
//ASPnetHost会从ASP.NET的执行应用程序域执行一个请求后返回流给当前web server所在的域,这实际上发生了一个域间的调用
      ASPnetHost.ProcessRequest (strRequestFile, ref output);//转换成字节流
      bs=System.Text.Encoding.Default.GetBytes (output);  
      iTotlaBytes=bs.Length ;  //调用套接字将执行结果返回  
      WriteHeader(httpver,"text/html",iTotlaBytes,"200 OK",ref sck);  
      FlushBuf(bs,ref sck);
     }
     else
     {try
      {
       fs=new FileStream( strRealFile,FileMode.Open,FileAccess.Read,FileShare.Read );
       BinaryReader reader=new BinaryReader(fs); //读取
       bs=new byte[fs.Length ];
       int rb;
       while((rb=reader.Read(bs,0,bs.Length ))!=0)
       {
 strResponse =strResponse +Encoding.Default.GetString(bs,0,rb);
        iTotlaBytes =iTotlaBytes+rb;
       }
       reader.Close();
       fs.Close();
     WriteHeader(httpver,"text/html",iTotlaBytes,"200 OK",ref sck);
       FlushBuf(bs,ref sck);
      }
      catch(System.IO.FileNotFoundException )
      {//假设找不到文件,报告404           WriteHeader(httpver,"text/html",iTotlaBytes,"404 OK",ref sck);
      }
     }
    }
    sck.Close(); //Http请求结束
   }
  }

// WriteHeader想客户端发送HTTP头
  public void WriteHeader(string ver,string mime,int len,string statucode,ref Socket sck)  {
   string buf="";

   if(mime.Length ==0)
   {
    mime="text/html";

    buf=buf+ver+ statucode + "/r/n";
    buf=buf+"Server:MyIIS"+"/r/n";
    buf=buf+"Content-Type:"+mime +"/r/n";
    buf=buf+"Accept-Rabges:bytes"+"/r/n";
    buf=buf+"Content-Length:"+ len +"/r/n/r/n";
    byte[] bs=Encoding.Default.GetBytes(buf);
    FlushBuf(bs,ref sck);
   }
  }

// FlushBuf刷新向客户发送信息缓冲区
  public void FlushBuf(byte[] bs,ref Socket sck)
  {
   int iNum=0;
   try
   {
    if(sck.Connected)
    {
     if((iNum=sck.Send(bs,bs.Length ,0))==-1)
     {
      Console.WriteLine("Flush Err:Send Data err");
     }
     else
     {
      Console.WriteLine("Send bytes :{0}",iNum);
     }
    }
    else
    {
     Console.WriteLine("Client diconnectioned!");
    }
   }
   catch(Exception e)
   {
    Console.WriteLine("Error:{0}",e);
   }
  }
 }

// ASPDOTNETHost类实例需要跨越两个应用程序域,所以继承自MarshalByRefObject
 class ASPDOTNETHost:MarshalByRefObject
 {
  public void  ProcessRequest( string fileName ,ref string output)
  {
   MemoryStream  ms=new MemoryStream(); //内存流,当然为了速度
   StreamWriter sw = new StreamWriter(ms);  //输出
   sw.AutoFlush = true;      //设为自动刷新 /先构造一个HttpWorkRequest请求类,以便ASP.NET能够分析获取请求信息,同时传入一个输出流对象供ASP.NET执行期间返回html流
   HttpWorkerRequest worker = new SimpleWorkerRequest( fileName, "" ,sw) ;   // 调度某个页,这里面的包含很多细节,后面分析
   HttpRuntime.ProcessRequest( worker ) ;
   StreamReader sr= new StreamReader(ms); //准备从内存流中读取
   ms.Position =0;     //移动指针到头
   output = sr.ReadToEnd();   
  }
}
}
HttpRuntime.ProcessRequest( worker ) ;包括了那些细节呢?大体上如下:
      1、首先,worker对象传入给ASP.NET的应用程序域,告知发生了对于哪一个ASPx文件的请求,以及当前目录是什么,如果在执行期间发生的输出内容应该写到哪里(sw对象)。这发生一个由web server当前应用程序域到我们自己建立的ASP.NET应用程序域的跨(应用程序)域调用,还可能由于是第一次访问,会发生了全局事件、或者session事件等。
      2、ASP.NET的应用程序域会检测请求的ASPx文件是否存在,不存在,就报错;如果存在还要看看代码缓存中是否存在上次编译的代码,如果存在且ASP.NET检测到不需要重新编译,会直接执行缓存中的代码;如果不存在或者代码过期需要重新编译,就需要读取ASPx文件,编译成.NET的代码,存入缓存。可能有些页存在代码和模板分离成多个文件,甚至包括一些资源文件,这些都需要读取后编译成.NET的虚拟机代码,然后在托管环境里执行。
      3、执行ASP.NET的编译代码缓存中的代码,输出数据利用sw对象输出。
    当然,根据不同的配置,还有很多方法的调用/事件的发生等细节不同。
    如何调试运行以上程序,观察结果呢?
    建立一个控制台类型工程,将上述代码录入后编译,将得到的程序拷贝在作为站点应用起始目录(譬如c:/inetpub/wwwroot/myweb)的bin子目录下,然后启动,这样在其中创建ASP.NET应用程序域才不会因为程序集加载失败而出错。建立一个asp.net工程在目录下,添加default.htm文件和测试用的test.aspx,加入.NET执行代码,然后启动IE,在地址栏分别输入:http://127.0.0.1:8001/default.htm http://127.0.0.1:8001/test.aspx感受一下执行过程。甚至你可以建立的工程中设定断点之类,仔细调试和观察其中的细节。亲手试一试吧,一定有收获的!

四、 自己构造ASP.NET宿主的意义
      费了半天劲搞自己的ASP.NET宿主,对于我们有何意义呢?
首先,是大致从代码级清楚分析ASP.NET执行细节,自己学习了解执行细节,除了可以在出现ASP.NET故障可以进行精确定位和排除外,还可以帮助我们在写ASP.NET应用程序时写出更有效率和健壮的代码。
      其次,我们可以提供一个思路,可以将我们的ASP.NET程序运行于低配置机器上,脱离IIS。ASP.NET的“原配”宿主IIS需要运行在Server OS上,要知道在安全专家眼中,IIS可是大隐患的源头之一。我们可以将很多传统程序利用ASP.NET编写,但脱离IIS独立执行,譬如在win98系统上执行ASP.NET。web server和ASP.NET都在托管环境中执行,相比较ISAPI建立宿主然后执行,除提高效率外,还可以使用.NET平台提供的丰富管理调控功能,写B/S程序更接近传统程序编写方式,这对于程序员来讲都是效率(编写代码的效率和执行效果效率)的保证。
另外,对于采用ASP.NET做的项目,大家可以很方便进行开发调试、运行维护、安装。即使是普通桌面程序,我们也可以通过类似制作网页的方式编写这些界面和代码,然后独立建立类似本例中的Host环境,根据用户交互请求加载执行某些页面,然后将界面在客户端通过相关组件显示出来。你可以通过此获得ASP.NET的即时编译功能和ASP.NET宿主托管环境,大量可自由使用的API,便于开发、安装、维护。毕竟,托管环境几乎准备了您需要的一切功能。

五、 参考资料
   1、.NET MSDN
   2、清华大学出版社《.NET网络高级编程》Andrew Krowczyk  Viond  Kumar原著
   3、清华大学出版社《.NET框架程序设计(修订版)》Jsfftry Richter著

 

  • 0
    点赞
  • 1
    评论
  • 0
    收藏
  • 一键三连
    一键三连
  • 扫一扫,分享海报

相关推荐
本博客采用ASP.NET2.0 c# + MSSQL2005 + XML + jquery + anthem.net开发,暂时为单用户版本,以下简单介绍... 程序基于类似petshop标准三层架构,方便维护与扩展(不过暂时只支持MSSQL数据库-_-||) 前台使用jquery框架实现完全无刷新,可以更快捷进行各种操作,并且交互结果能即时显示,无需刷新页面 边栏实现模块化,除导航菜单外所有功能模块都可以在后台自由排序删除加载,并且可以自定义新模块 后台因安全原因使用服务器端anthem.net框架开发,实现数据操作无刷新 所有数据操作全部封装在数据库里,一定程度上提高了性能 无刷新动态换肤,并支持自定义皮肤样式 系统暂时支持ie7、firefox+、maxthon、腾讯tt、theworld几款浏览器,ie6有些小问题(主要是ie6自身bug) v1.080709更新记录: 重写后台页面结构,改进了后台一些功能 修正了一些外部类库写法及错误注释 前台采用最新jquery1.2.6版,后台编辑器采用最新3.4.8版 增加文章内容判断功能,即如果内容快照为空,则不折叠显示文章内容(适合篇幅较短文章) 改进ajax功能,优化了一些js写法 数据库下载改用实体文件,直接附加即可 v1.080606更新记录: 数据库改动较大,个别表增加字段,大部分存储过程重写,增加sql2005分页存储过程 增加前台评论分页,重写了分页机制以及底层一些函数(因为数据库改动) 增加日志保存草稿功能,并增加日志转移功能 增加相片自定义水印功能 改进后台评论查看机制,增加后台统计功能 集成gzip压缩技术(性能是否提升还要进一步考证- -||) 增强自定义属性,如默认css、分页大小、水印等等 增加一款皮肤:仿msdn红色 修改了关于css列表加载顺序bug,增强对ie6支持(不过仍有些小问题- -||) 首页结构有小修改,并修正了一些小问题 v1.080516更新记录: 增加相片生成缩略图功能 增加生成静态页功能(目前只针对日志) 增加ttings提供一款皮肤:ttings 修正一些小bug 注: * 程序因注重代码结构层次和可读性,牺牲了些许性能,request会比较多 * gzip压缩性能有待进一步测试,因为虽然压缩后减少了数据传输量,但压缩本身要消耗一定服务器资源,不知孰重孰轻... * 文本编辑器采用是开源DotNetTextBox 【数据库配置】 web/App_Data里是数据库实体文件,用mssql2005(必须是mssql2005)附加即可 然后修改程序web.config数据库连接字符串,有两个地方,一个在<appSettings>节点内,一个在<connectionStrings>节点内,这个就不多说了 【管理员帐户配置】:默认帐号/密码:51aspx/51aspx 也可以运行web/tools下Encrypt.aspx,输入自定义用户名和密码,选md5,加密字符串,然后打开web.config,找到 <add key="AdmID" value="*************************"/> <add key="AdmPWD" value="*************************"/> 分别替换value值,完成后就可以登录后台了,后台登录页是adm/login.aspx;index.htm是首页
自由宿主后台管理系统是一套通用后台管理系统,之所以给他加上“宿主”2个字,是因为这套系统本身没有自己数据库,而必须依赖于其他数据库才能运行,自由指系统灵活性强,能够根据数据库表结构做一些简单配置后系统就能够实现想要功能。简单说,就是系统没有根据特定数据库来设计,你给他套上什么结构数据库表,他就做什么样事情。   自由宿主Asp.net网站通用后台管理系统特点如下: 1. 后台本身没有数据库,自己设计数据库后,与后台连接,配置好节点后无需二次开发,直接就可以实现对数据库增删改查功能; 2. 后台密码支持自定义加密/解密算法,可以在设置登录表时候进行配置; 3. 后台编辑界面可以根据配置自动生成编辑器,同时会自动生成验证脚本; 4. 后台支持多表关联,列表查询时候会自动根据关联字段显示相应内容; 5. 兼容IE6 ,火狐浏览器等主流浏览器; 6. 对数据库表操作,如增删改查操作只需要进行简单设置即可完成,无需再进行任何编码实现。   自由宿主Asp.net网站通用后台管理系统3.2正式版更新功能: 1.修正edit.js内容验证无效bug; 2.添加sqliste数据库支持; 3.修正编辑后返回列表丢失删除功能Bug; 4.Main页面菜单显示改为json对象传输到前台,然后由menu.js脚本实现菜单展示功能,方便二次开发对界面修改,菜单可以控制允许一次展开多个或者只允许展开一个; 5.修正图片上传后缀名区分大小写问题; 6.系统生成字符类型主键修改为GUID,主键字段长度至少需要为32位字符; 7.上传文件保存文件名修改为GUID,数据库相印字段长度至少需要为32位字符; 8.配置页面数据库表增加视图显示; 9.增加省市县联动控件(包括省市县3级联动,省市2级联动和省市(县级可选)联动),通过使用Request[“_tzhp_字段名称”]获取省区划编码,Request[“_tzhc_字段名称”]获取市区划编码,Request[“_tzhr_字段名称”]获取区县区划编码,使用Request[“字段名称”]获取选中最后级区划编码(写入数据库值)。Area.js脚本为全国省市县名称和区划编码,其中_codeToObject(code)方法可以将区划编码转为区划对象,格式为: { p: {Code, Name, Level}, c: {Code, Name, Level}, r{Code, Name, Level}},其中p为省份信息,c为城市信息,r为区县信息; 10.在授权模式下,列表显示字段如为图片方式上传,则可以直接显示图片,区划字段将显示为对应区划名称,没有授权情况下原样输出; 11.在授权模式下,标题栏将不显示“自由宿主xx”字样; 12.在授权模式下超级管理员后台可以通过配置文件增加功能链接; 13.tzh_tableInfo 和tzh_menuInfo表增加字段“strTbGroup varchar(32)”旧版本升级程序请手动在数据库中添加此列; 14.修正使用Access数据库多个表存在关联且有重复字段名情况下可能出现列表数据为空情况;   数据库升级方式请参见 http://www.tzhtec.com/art/6f456394186.aspx
自由宿主后台管理系统是一套通用后台管理系统,之所以给他加上“宿主”2个字,是因为这套系统本身没有自己数据库,而必须依赖于其他数据库才能运行,自由指系统灵活性强,能够根据数据库表结构做一些简单配置后系统就能够实现想要功能。简单说,就是系统没有根据特定数据库来设计,你给他套上什么结构数据库表,他就做什么样事情。 自由宿主Asp.net网站通用后台管理系统特点如下: 1. 后台本身没有数据库,自己设计数据库后,与后台连接,配置好节点后无需二次开发,直接就可以实现对数据库增删改查功能; 2. 后台密码支持自定义加密/解密算法,可以在设置登录表时候进行配置; 3. 后台编辑界面可以根据配置自动生成编辑器,同时会自动生成验证脚本; 4. 后台支持多表关联,列表查询时候会自动根据关联字段显示相应内容; 5. 兼容IE6 ,火狐浏览器等主流浏览器; 6. 对数据库表操作,如增删改查操作只需要进行简单设置即可完成,无需再进行任何编码实现。 自由宿主Asp.net网站通用后台管理系统3.0正式版更新功能: 1. 调整列表和编辑界面样式,列表修改为采用easyuigrid显示; 2. 编辑页面**期、编辑器和验证脚本整理,调整到脚本文件edit.js中,如果有其他需求,可以修改该脚本文件; 3. 增加oracle和mysql数据库支持,支持,系统可以支持ms sql、access、oracle和mysql等主流数据库; 4. 由于原来jquery.ui在系统上使用并不多,在本次改版后去除了jquery.ui,css和js上有大量瘦身; 5. 登陆后主页显示default.aspx页面修改允许用户自行修改设置; 6. 系统对**台要求提升,最低运行环境为framework3.5,不再对framework2.0进行支持; 7. 对SQL查询语句做了优化; 8. 系统架构重新调整;
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值