gpt4 book ai didi

c# - 为什么类型 System.__ComObject 声称(有时)是公开的,而实际上不是?

转载 作者:IT王子 更新时间:2023-10-29 04:39:34 25 4
gpt4 key购买 nike

当我出于好奇而反射(reflection)所有类型以检查其他内容时,我偶然发现了一个奇怪的现象。

为什么程序集 mscorlib.dll(有时?)的类 System.__ComObject 声称是公开的,而实际上它似乎是非公开的?如果我在一个简单的 C# 控制台应用程序中运行以下代码:

var t = Type.GetType("System.__ComObject");
Console.WriteLine(t.IsPublic); // "True" ?!
Console.WriteLine(t.IsVisible); // "False"

输出似乎是矛盾的。非嵌套类型(t.IsNested 为假)应该为 IsPublicIsVisible 提供相同的真值。当我使用 IL DASM 查看程序集时,我看到:

.class private auto ansi beforefieldinit System.__ComObject
extends System.MarshalByRefObject
{
} // end of class System.__ComObject

在我看来,它看起来很像一个非公共(public)类,对应于下面的 C# 代码:

namespace System
{
// not public
internal class __ComObject : MarshalByRefObject
{
...
}
}

当我与另一个具有相似名称 System.__Canon 和相似 IL 修饰符的类型进行比较时,IsPublicIsVisible 都返回如预期的那样错误。

有谁知道为什么(以及何时)Type.GetType("System.__ComObject").IsPublic 给出 true?

最佳答案

Mono's implementation __ComObject 可能会提供一些线索。它确实被声明为 internal,但评论说 “它没有公共(public)方法,它的功能通过 System.Runtime.InteropServices.Marshal 公开”。我没有深入研究 Marshal,但我认为它负责 GetType() 的实现,因此它可以自定义 IsPublic 属性也是。

根据我的经验,Type.GetType("System.__ComObject").IsPublic 始终是 true。关于 GetType("System.Net.Mail.MSAdminBase"),我相信它是通过 customized primary interop assembly 公开的 COM 类,其中可以显式控制类型的可见性(尽管这只是我的想法,尚未进行任何研究)。

[更新]

我有 latest Framework source code并发现我错误地假设 __ComObject 类型的 IsPublic 属性的自定义是由 Marshal 完成的。实际上,它是由非托管代码完成的。 Object.GetType() 在 Object.cs 中定义如下:

[MethodImplAttribute(MethodImplOptions.InternalCall)]
public extern Type GetType();

其在 .NET 4.x 中实现的非托管源代码不可用,但它是 available对于.NET 2.0。有一个 excellent answer关于 .NET 2.0 中的 Object.GetType 实现。我只想补充一点,IsPublicSystem.Runtime.InteropServices._Type 接口(interface)定义,System.Type 派生自该接口(interface),并且可以被覆盖通过任何 System.Type-descendent 类。显然,此类的实现由 Object.GetType 返回,并且发生在 ObjectNative::GetClass (sscli20\clr\src\vm\comobject.cpp) 中,如图the answer :

if (!objRef->IsThunking())
refType = typeHandle.GetManagedClassObject();
else
refType = CRemotingServices::GetClass(objRef);

我确认 IsPublic 的行为取决于框架版本。我在不同的虚拟机中尝试了以下 PowerShell 脚本:

Write-Host ([System.Environment]::Version) ([Type]::GetType("System.__ComObject")).IsPublic

输出是:

2.0.50727.3643 False (WinXP)
4.0.30319.18052 True (Win7)
4.0.30319.19079 True (Win8)

显然,自 .NET 2.0 以来,它已从 False 更改为 True。我同意 True__ComObject(internal class __ComObject ...)的声明不一致。我个人认为没有理由进行这种更改,因为获取 __ComObject 实例的唯一方法是通过互操作。

关于c# - 为什么类型 System.__ComObject 声称(有时)是公开的,而实际上不是?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18090412/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com