我有一个名为Assembly1的F#程序集
我有以下模块:
namespace Assembly1
module MyModule =
type MyClass(x: int, y: int) =
new() = MyClass(0, 0)
在引用此F#程序集的C#程序集中,以下代码为我提供了值' null'回:
var myType = Type.GetType("Assembly1.MyModule.MyClass, Assembly1");
我想做的不可能吗?
答案 0 :(得分:6)
要添加到Mark's answer,还值得注意的是,F#中的许多模块由IL(因此非F#语言)中的不同名称表示,而不是F#本身中显示的。
例如,这段代码:
open System
open System.Reflection
open Microsoft.FSharp.Reflection
let private core =
AppDomain.CurrentDomain.GetAssemblies()
|> Seq.find (fun a -> a.GetName().Name = "FSharp.Core")
let private seqMod =
core.GetTypes()
|> Seq.filter FSharpType.IsModule
|> Seq.find (fun t ->
t.FullName = "Microsoft.FSharp.Collections.SeqModule")
将在FSharp.Core中找到Seq模块。
FSharp.Reflection命名空间有一堆辅助方法,可以使用System.Reflection来处理F#特定类型,但是在FSI中加载几个程序集并且如果你在那里玩那些组件是值得的。我会用F#代码进行大量的反思工作。
您可以使用CompiledName属性自行创建具有“不匹配”名称的模块 - 这对于您希望类型和模块具有相同名称的时间尤其有用。常规约定(如Seq类型/模块所示)是使用Compiled name注释模块。
[<CompiledName("MyTypeModule")>]
module MyType =
let getString m =
match m with
| MyType s -> s
答案 1 :(得分:5)
由于您已将MyClass
放入模块中,因此将其编译为嵌套类。它的名字是Assembly1.MyModule+MyClass
。
从C#开始,您应该可以像这样加载它:
var myType = Type.GetType("Assembly1.MyModule+MyClass, Assembly1");
如果您不想拥有嵌套类(在C#中通常不赞成),您可以直接在命名空间中定义它:
namespace Assembly1.LibraryXyz
type MyClass(x: int, y: int) = class end
此课程的名称为Assembly1.LibraryXyz.MyClass
。