3

我正在开发一个新的 WCF Web 服务,该服务将托管在现有的 ASP.NET Web 应用程序中。当我尝试为该服务运行 .svc 文件时,我收到一个异常,它找不到程序集的文件。

无法加载文件或程序集“System.IdentityModel,Version=4.0.0.0,Culture=neutral,PublicKeyToken=b77a5c561934e089”或其依赖项之一。该系统找不到指定的文件。

问题是,这个服务似乎没有在任何地方引用这个(特定的)程序集。有一个加载 System.IdentityModel 的引用项目,但它引用版本 3.0.0.0。事实上,所有这些项目都针对 3.5 运行时,并且在任何地方都没有对任何 4.0 程序集的引用。

这是<compilation>Web 应用程序的 Web.config 中的标记:

<compilation debug="true">
  <assemblies>
    <clear/>
    <add assembly="System.Design, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Management, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Data.OracleClient, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System.Configuration.Install, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Transactions, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System.Messaging, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Data, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System.Drawing, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Xml, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System.Configuration, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.EnterpriseServices, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Web, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Web.Services, Version=2.0.0.0, Culture=neutral, PublicKeyToken=B03F5F7F11D50A3A"/>
    <add assembly="System.Core, Version=3.5.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System.Web.Extensions, Version=3.5.0.0, Culture=neutral, PublicKeyToken=31BF3856AD364E35"/>
    <add assembly="System.Xml.Linq, Version=3.5.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
    <add assembly="System.Data.DataSetExtensions, Version=3.5.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>
  </assemblies>
</compilation>

注意:该<clear />标签是后来作为故障排除尝试添加的。它似乎对行为没有任何影响。我添加它是因为 IIS 配置管理器的“.NET 编译”选项卡给了我一个错误,即 System.Data 被添加了两次,但显然不在此文件中。添加<clear />为我处理了它,但我在 Machine.config 中找不到有问题的重复项,并且没有基础网站 Web.config,所以我想知道是否可能有某个地方我没有找到可以成为问题的原因。

服务器正在运行带有 .NET 3.5 SP1 的 Server 2008。我在本地 IIS 安装上遇到了同样的问题(虽然System.Configuration不是 IdentityModel,但仍然......),它是在带有 .NET 4.0 的 Windows 7 x64 上。

有谁知道它为什么试图加载这个程序集(当只引用版本 3.0.0.0 时)以及我能做些什么来纠正它?

4

1 回答 1

1

我弄清楚了这个问题,并认为它可能对可能遇到同样问题的其他人有所帮助。

当一个 WCF 服务被激活时,部分流程Type.GetType(string)用来获取代表服务契约实现的类型。因为Type.GetType(string)要求目标类型是:

  • 在调用程序集中(在这种情况下,它永远不会)
  • 在 mscorlib 中(同样,它永远不会)
  • 使用程序集限定名称完全指定

除非使用其程序集限定名称指定类型,否则它将失败。

正因为如此(并且为了仅使用命名空间限定名称来更轻松地指定服务类型),激活器将扫描所有引用的程序集并调用Assembly.GetType(string)(它可以只采用命名空间限定名称),直到它找到兼容的类型。

虽然我无法解释 4.0 引用的具体来源,但它一定是在引用的程序集中。通过ServiceHost将 .svc 文件中的标记更改为我的服务使用程序集限定名称,WCF 成功加载了服务。

于 2010-08-06T14:37:16.120 回答