WEB开发网
开发学院WEB开发ASP.NET Asp.net 下把自己的程序集加入到 GAC 阅读

Asp.net 下把自己的程序集加入到 GAC

 2009-12-04 16:52:02 来源:WEB开发网   
核心提示:因为服务器的性能有限,发现自从使用了 Castle + Nhibernate 之后,Asp.net 下把自己的程序集加入到 GAC,服务器在更新web站点之后的重新编译过程特别漫长,有时候都超过了1分钟,知道 <compilation>中有 <assemblies>的定义,增加配置如下:view
因为服务器的性能有限,发现自从使用了 Castle + Nhibernate 之后,服务器在更新web站点之后的重新编译过程特别漫长,有时候都超过了1分钟,尽管发布的web程序也是编译好的。下面是web使用的动态库:

Castle.Core.dll
Castle.DynamicPRoxy2.dll
Castle.Facilities.AutomaticTransactionManagement.dll
Castle.Facilities.NHibernateIntegration.dll
Castle.MicroKernel.dll
Castle.Services.Transaction.dll
Castle.Windsor.dll
FredCK.FCKeditorV2.dll
Iesi.Collections.dll
log4net.dll
MySQL.Data.dll/System.Data.SQLite.dll
NHibernate.dll
UrlRewritingNet.UrlRewriter.dll

每次我的web项目有更新重新编译后生成的web.dll上传到服务器,服务器w3p就会重启,csc我的web目录下的文件,整个过程真是漫长的无法忍受。

[不知道各位大侠有没有什么好的建议]

后来想到把上面的那些动态库放入 GAC 中,会不会减少重启需要时间(实践证明没有明显的变化)。

本文没有解决当初的问题,但是因为涉及到一个在 asp.net 中引用存放在GAC中的第三方动态库的问题,所以还是记录下来(:-)这才是本文重点)。

上面所有的动态库都是签过名的,放入GAC的过程:

1. 注册第三方动态库入GAC

gacutil -i Castle.Core.dll
gacutil -i Castle.DynamicProxy2.dll
gacutil -i Castle.Facilities.AutomaticTransactionManagement.dll
gacutil -i Castle.Facilities.NHibernateIntegration.dll
gacutil -i Castle.MicroKernel.dll
gacutil -i Castle.Services.Transaction.dll
gacutil -i Castle.Windsor.dll
gacutil -i FredCK.FCKeditorV2.dll
gacutil -i Iesi.Collections.dll
gacutil -i Intelligencia.UrlRewriter.dll
gacutil -i log4net.dll
gacutil -i MySql.Data.dll
gacutil -i NHibernate.dll
gacutil -i System.Data.SQLite.dll
gacutil -i UrlRewritingNet.UrlRewriter.dll

这个过程比较容易,gacutil 可以在 .net framework 的 sdk 下找到(再分发包)

2. 重新定义web项目对第三方动态库的引用方式:直接引用 GAC

这里遇到另外一个问题, vs2005中无法引用到自己定义到GAC的第三方动态库,除非做如下设定:

假设 自己定义的第三方动态库存放于: c:\3rdlibs

修改注册表:HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\.NETFramework\AssemblyFolders

在这个项目下加入一个项,并指向 c:\3rhlibs

现在可以在我们的web项目中引用第三方的GAC库了。

3.重新编译web,调试:

第一个问题:配置错误
view plaincopy to clipboardprint?
配置错误  
说明: 在处理向该请求提供服务所需的配置文件时出错。请检查下面的特定错误详细信息并适当地修改配置文件。  
 
分析器错误信息: 未能加载文件或程序集“UrlRewritingNet.UrlRewriter”或它的某一个依赖项。系统找不到指定的文件。 (web.config line 50)  
 
源错误:  
 
行 48:    
行 49:   <httpModules>  
行 50:    <add name="UrlRewriteModule" type="UrlRewritingNet.Web.UrlRewriteModule, UrlRewritingNet.UrlRewriter"/>  
  
 
源文件: web.config  行: 50  
配置错误
说明: 在处理向该请求提供服务所需的配置文件时出错。请检查下面的特定错误详细信息并适当地修改配置文件。

分析器错误信息: 未能加载文件或程序集“UrlRewritingNet.UrlRewriter”或它的某一个依赖项。系统找不到指定的文件。 (web.config line 50)

源错误:

行 48:   
行 49:   <httpModules>
行 50:    <add name="UrlRewriteModule" type="UrlRewritingNet.Web.UrlRewriteModule, UrlRewritingNet.UrlRewriter"/>


源文件: web.config  行: 50





显然我们的web项目没有找到已经存放在 GAC 中的库。放入GAC以前这些库是跟随编译输出到 web/bin 目录下的。现在自动定位不到,我们可以修改 web.config 来指定库的引用方式:

view plaincopy to clipboardprint?
<configuration>  
 ......  
 <runtime>  
  <assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">  
   <qualifyAssembly partialName="log4net" fullName="log4net, Version=1.2.10.0, Culture=neutral, PublicKeyToken=1b44e1d426115821"/>  
   <qualifyAssembly partialName="Castle.Windsor" fullName="Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc"/>  
   <qualifyAssembly partialName="Castle.MicroKernel" fullName="Castle.MicroKernel, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc"/>  
   ... ...  
  </assemblyBinding>  
 </runtime>  
</configuration> 
<configuration>
 ......
 <runtime>
  <assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
   <qualifyAssembly partialName="log4net" fullName="log4net, Version=1.2.10.0, Culture=neutral, PublicKeyToken=1b44e1d426115821"/>
   <qualifyAssembly partialName="Castle.Windsor" fullName="Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc"/>
   <qualifyAssembly partialName="Castle.MicroKernel" fullName="Castle.MicroKernel, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc"/>
   ... ...
  </assemblyBinding>
 </runtime>
</configuration>

再次编译调试运行

第二个错误:编译错误
view plaincopy to clipboardprint?
“/”应用程序中的服务器错误。  
--------------------------------------------------------------------------------  
 
编译错误  
说明: 在编译向该请求提供服务所需资源的过程中出现错误。请检查下列特定错误详细信息并适当地修改源代码。  
 
编译器错误信息: CS0012: 类型“Castle.Windsor.IContaineraccessor”在未被引用的程序集中定义。必须添加对程序集“Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc”的引用。  
 
源错误:  
  
 
[没有相关的源行]  
  
 
源文件: c:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\Temporary ASP.NET Files\root\xxxxxx\yyyyyyyy\App_Web_default.aspx.zzzzzzz.aaaaaaaa.0.cs  行: 133  
“/”应用程序中的服务器错误。
--------------------------------------------------------------------------------

编译错误
说明: 在编译向该请求提供服务所需资源的过程中出现错误。请检查下列特定错误详细信息并适当地修改源代码。

编译器错误信息: CS0012: 类型“Castle.Windsor.IContainerAccessor”在未被引用的程序集中定义。必须添加对程序集“Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc”的引用。

源错误:


[没有相关的源行]


源文件: c:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\Temporary ASP.NET Files\root\xxxxxx\yyyyyyyy\App_Web_default.aspx.zzzzzzz.aaaaaaaa.0.cs  行: 133


这次是一个编译错误,但是错误的信息并不清晰,暂时无法定位到具体内容,修改 web.config, 增加一个配置节(compilation),如下

view plaincopy to clipboardprint?
<configuration>  
 <system.web>  
 ... ...   
  <compilation debug="true" />  
 ... ...  
 </system.web>  
</configuration> 
<configuration>
 <system.web>
 ... ... 
  <compilation debug="true" />
 ... ...
 </system.web>
</configuration>

再次调试:

view plaincopy to clipboardprint?
编译错误  
说明: 在编译向该请求提供服务所需资源的过程中出现错误。请检查下列特定错误详细信息并适当地修改源代码。  
 
编译器错误信息: CS0012: 类型“Castle.Windsor.IContainerAccessor”在未被引用的程序集中定义。必须添加对程序集“Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc”的引用。  
 
源错误:  
 
行 134:    }  
行 135:      
行 136:    protected ASP.global_asax applicationInstance {  
行 137:      get {  
行 138:        return ((ASP.global_asax)(this.Context.ApplicationInstance));  
  
 
源文件: c:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\Temporary ASP.NET Files\root\xxxxxx\yyyyyyyy\App_Web_default.aspx.zzzzzzz.aaaaaaaa.0.cs  行: 136  
编译错误
说明: 在编译向该请求提供服务所需资源的过程中出现错误。请检查下列特定错误详细信息并适当地修改源代码。

编译器错误信息: CS0012: 类型“Castle.Windsor.IContainerAccessor”在未被引用的程序集中定义。必须添加对程序集“Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc”的引用。

源错误:

行 134:    }
行 135:    
行 136:    protected ASP.global_asax ApplicationInstance {
行 137:      get {
行 138:        return ((ASP.global_asax)(this.Context.ApplicationInstance));


源文件: c:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\Temporary ASP.NET Files\root\xxxxxx\yyyyyyyy\App_Web_default.aspx.zzzzzzz.aaaaaaaa.0.cs  行: 136


这个错误信息最重要:

编译器错误信息: CS0012: 类型“Castle.Windsor.IContainerAccessor”在未被引用的程序集中定义。必须添加对程序集“Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc”的引用。

显然,我们在修改 web.config之后,iis服务重启了当前的web项目,并且对它进行了编译,首先被编译的就是 global.asax.cs, 其中某个初始化的变量,也可能是一个静态的变量,调用了 Castle.Windsor,并且 csc 找不到这个 Castle.Windsor ,真是一个麻烦的过程。

第一个尝试: 在web项目中的引用部分,对 Castle.Windsor 做设定,原来直接引用动态库的时候是同步输出到 web/bin 下的,引用 GAC之后此输出就取消了,现在重新设定为“复制到本地”。当然,设定后的编译会把此文件:Castle.Windsor输出到 web/bin下,如此一来,调试也可以通过了,因为他找到了这个动态库。

但是这并没有解除我们的疑惑,那就是, <runtime><assemblyBinding>... ...</assemblyBinding><runtime> 中定义的Assembly并不对编译过程发生作用,那么在编译过程中要用到的是那个节呢?

第二个尝试:想到第一个尝试中的 (compilation) 节,查看msdn的说明,知道 <compilation>中有 <assemblies>的定义,增加配置如下:

view plaincopy to clipboardprint?
<compilation debug="false">  
 <assemblies>  
  <add assembly="Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc"/>  
 </assemblies>  
</compilation> 
<compilation debug="false">
 <assemblies>
  <add assembly="Castle.Windsor, Version=1.0.3.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc"/>
 </assemblies>
</compilation>


然后取消web项目中对 Castle.Windsor 的 “复制到本地”,重新编译运行。

All are Working Fine.

到这里我们已经成功的把所以第三方引用的动态库放入到 GAC 中了

Tags:Asp net 自己

编辑录入:爽爽 [复制链接] [打 印]
赞助商链接