3

我有一个与此问题答案中提供的代码相关的问题。

CompilerParameters我遇到的问题是在运行时找不到传递给的三个引用程序集(System.dll、FSharp.Core.dll、FSharp.Powerpack.dll) 。我得到的错误是:

未知文件(0,0):错误 0:错误 FS0218:无法读取程序集 'c:\user s\utente\documents\visual studio 2010\Projects\TrashSolution\TrashSolution\bin\D ebug\FSharp.Core.dll '

如何告诉编译器在 GAC 中而不是项目的 bin 目录中搜索这些程序集?如果我在作为字符串提供的代码中打开命名空间,我如何知道要添加哪些程序集?我在哪里可以获得这些信息?

4

1 回答 1

7

在您链接的答案的代码中,底部有一行:

let asm = Reflection.Assembly.LoadFrom(fileinfo.Value.FullName)

如果您Reflection.Load改为调用并将完全限定的程序集名称传递给它,它将尝试从 GAC(以及其他一些地方,如果程序集不在 GAC 中)加载程序集。

let asm =
    Assembly.Load "SampleAssembly, Version=1.0.2004.0, Culture=neutral, PublicKeyToken=8744b20f8da049e3"

如果您不知道完全限定的程序集名称,则必须使用AssemblyName程序集的简单名称创建一个,然后调用采用 a而不是 a的Reflection.Load重载。AssemblyNamestring

let asmName = AssemblyName "Your.Assembly.Name"
let asm = Assembly.Load asmName

至于知道要加载哪些程序集——我认为没有一种简单的方法可以以编程方式确定它。我现在唯一能想到的两个解决方案:

  1. 如果您对给出的代码(作为字符串)有所FSharpCodeProvider了解,则可以使用 解析它并查看打开了哪些命名空间/模块以及使用了哪些类型。如果您想查看是否使用了某个特定的命名空间或类型(即,在编译代码时需要包含程序集引用),您可以创建一个.fsx命名空间的 Map(在您正在编译的地方)和/或将名称键入程序集名称并使用它来引用适当的程序集。
  2. 您可以“蛮力”搜索 GAC,方法是使用半文档化的 Fusion API 枚举安装在 GAC 中的所有程序集,然后使用反射检查每个程序集并确定它是否是您需要的。这可能非常慢,所以我会不惜一切代价避免它。如果您决定走这条路,您还必须使用该Assembly.ReflectionOnlyLoad方法来加载程序集!这允许在您完成检查后卸载程序集 - 如果您使用普通反射,则无法卸载程序集,并且您的程序可能会因OutOfMemoryException类似或类似情况而崩溃。

编辑:事实证明,通过其简单名称加载程序集是成功的,fsi而不是在普通的 F# 代码中,因为会自动为AppDomain.AssemblyResolve事件fsi安装一个处理程序。此事件由 CLR 在您尝试加载程序集且无法解析时触发;该事件为您提供了一种“手动”解析程序集和/或动态生成程序集并将其返回的方法。

如果您FileNotFoundException尝试在 F# 项目中运行代码时查看引发的问题,您将在异常的 Fusion Log 属性中看到类似的内容:

=== Pre-bind state information ===
LOG: User = Jack-Laptop\Jack
LOG: DisplayName = System
 (Partial)
WRN: Partial binding information was supplied for an assembly:
WRN: Assembly Name: System | Domain ID: 1
WRN: A partial bind occurs when only part of the assembly display name is provided.
WRN: This might result in the binder loading an incorrect assembly.
WRN: It is recommended to provide a fully specified textual identity for the assembly,
WRN: that consists of the simple name, version, culture, and public key token.
WRN: See whitepaper http://go.microsoft.com/fwlink/?LinkId=109270 for more information and common solutions to this issue.
LOG: Appbase = file:///C:/Users/Jack/Documents/Visual Studio 2010/Projects/StackOverflow1/StackOverflow1/bin/Debug/
LOG: Initial PrivatePath = NULL
Calling assembly : StackOverflow1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null.
===
LOG: This bind starts in default load context.
LOG: No application configuration file found.
LOG: Using host configuration file: 
LOG: Using machine configuration file from C:\Windows\Microsoft.NET\Framework64\v4.0.30319\config\machine.config.
LOG: Policy not being applied to reference at this time (private, custom, partial, or location-based assembly bind).
LOG: Attempting download of new URL file:///C:/Users/Jack/Documents/Visual Studio 2010/Projects/StackOverflow1/StackOverflow1/bin/Debug/System.DLL.
LOG: Attempting download of new URL file:///C:/Users/Jack/Documents/Visual Studio 2010/Projects/StackOverflow1/StackOverflow1/bin/Debug/System/System.DLL.
LOG: Attempting download of new URL file:///C:/Users/Jack/Documents/Visual Studio 2010/Projects/StackOverflow1/StackOverflow1/bin/Debug/System.EXE.
LOG: Attempting download of new URL file:///C:/Users/Jack/Documents/Visual Studio 2010/Projects/StackOverflow1/StackOverflow1/bin/Debug/System/System.EXE.

查看该日志的底部,您将看到 CLR 在放弃之前搜索程序集的位置。

这是一个简单的处理程序,可让您了解如何使用 AppDomain.AssemblyResolve 处理程序手动解析程序集。(注意:需要在尝试加载程序集的代码之前添加处理程序!)

System.AppDomain.CurrentDomain.add_AssemblyResolve (
    System.ResolveEventHandler (fun _ args ->
        let resolvedAssembly =
            System.AppDomain.CurrentDomain.GetAssemblies ()
            |> Array.tryFind (fun loadedAssembly ->
                // If this assembly has the same name as the one we're looking for,
                // assume it's correct and load it. NOTE : It may not be the _exact_
                // assembly we're looking for -- then you'll need to adjust the critera below.
                args.Name = loadedAssembly.FullName
                || args.Name = loadedAssembly.GetName().Name)

        // Return null if the assembly couldn't be resolved.
        defaultArg resolvedAssembly null))

如果您将该代码添加到新的 F# 控制台项目,然后添加AssemblyName与 Assembly.Load 一起使用的代码,您应该能够加载System程序集,因为它在 F# 项目中默认被引用,并且会在您运行时加载项目。如果您尝试解析System.Drawing,它将失败,因为我们的自定义事件处理程序找不到程序集。显然,如果您需要一些更复杂的程序集解析逻辑,您应该以任何对您的应用程序有意义的方式将其构建到事件处理程序中。

最后,这是异常消息中提到的 MSDN 白皮书的链接:Best Practices for Assembly Loading。如果您遇到困难并且无法弄清楚如何解决您需要的程序集,那么值得一读。

于 2013-01-29T11:46:26.877 回答