如何通过反射来确定对象的类型是由我自己的程序集或.NET Framework中的类定义的?检测对象的类型是否是由.NET Framework定义的类型
我不想在代码中提供我自己的程序集的名称,因为它应该可以用于任何程序集和名称空间。
如何通过反射来确定对象的类型是由我自己的程序集或.NET Framework中的类定义的?检测对象的类型是否是由.NET Framework定义的类型
我不想在代码中提供我自己的程序集的名称,因为它应该可以用于任何程序集和名称空间。
第三方类型在哪里?您可能想区分声称由Microsoft提供的类型和不提供的类型。
using System;
using System.Linq;
using System.Reflection;
class Test
{
static void Main()
{
Console.WriteLine(IsMicrosoftType(typeof(string)));
Console.WriteLine(IsMicrosoftType(typeof(Test)));
}
static bool IsMicrosoftType(Type type)
{
object[] attrs = type.Assembly.GetCustomAttributes
(typeof(AssemblyCompanyAttribute), false);
return attrs.OfType<AssemblyCompanyAttribute>()
.Any(attr => attr.Company == "Microsoft Corporation");
}
}
当然,任何类型的可能要求成为微软一个给定的这个方案,但如果你实际上只去调用它自己的类型和框架的,我怀疑这应该能正常运行。
或者,您可以使用程序集的公钥标记。这很可能很难伪造。它依靠微软为他们所有的程序集使用公共公钥,而他们不这么做(根据Mehrdad的评论)。但是,您可以轻松地将此解决方案适用于集合的接受“这是来自Microsoft”的公钥。也许在某种程度上将二者结合起来的方法和报告进行进一步的检查任何差异...
static bool IsMicrosoftType(Type type)
{
AssemblyName name = type.Assembly.GetName();
byte[] publicKeyToken = name.GetPublicKeyToken();
return publicKeyToken != null
&& publicKeyToken.Length == 8
&& publicKeyToken[0] == 0xb7
&& publicKeyToken[1] == 0x7a
&& publicKeyToken[2] == 0x5c
&& publicKeyToken[3] == 0x56
&& publicKeyToken[4] == 0x19
&& publicKeyToken[5] == 0x34
&& publicKeyToken[6] == 0xe0
&& publicKeyToken[7] == 0x89;
}
并非所有的Microsoft程序集都具有相同的公钥令牌(即使是现在)。 – 2009-06-07 20:35:26
Ick。将更新答案。 – 2009-06-07 20:58:24
obj.GetType().Assembly == System.Reflection.Assembly.GetExecutingAssembly()
检查类型是否在当前程序集中声明。
与Mehrdad的答案类似,但即使代码在某些其他应用程序中执行,也允许进行相同的检查。
obj.GetType().Assembly == typeof(SomeTypeYouKnowIsInYourAssembly).Assembly
+1这是一个聪明的做法。 – user7116 2009-06-07 21:12:36
基于乔恩的答案,迈赫达德的评论,似乎以下三个值用于公钥标记(从AssemblyName.FullName ),用于从.NET 2.0和更高.NET Framework提供组件:
公钥= b77a5c561934e089
公钥= b03f5f7f11d50a3a
公钥= 31bf3856ad364e35
这是从下面的代码生成:
private void PrintAssemblyInfo(string fullName)
{
string[] parts = fullName.Split(',');
Console.WriteLine(" - {0}, {1}", parts[0], parts[3]);
}
private void GenerateInfo(string path)
{
foreach (var file in Directory.GetFiles(path,
"*.dll",
SearchOption.AllDirectories))
{
try
{
Assembly assembly = Assembly.ReflectionOnlyLoadFrom(file);
PrintAssemblyInfo(assembly.GetName().FullName);
}
catch { }
}
}
private void GenerateInfo()
{
GenerateInfo(@"C:\Windows\Microsoft.NET\Framework\v2.0.50727");
GenerateInfo(@"C:\Windows\Microsoft.NET\Framework\v3.0");
GenerateInfo(@"C:\Windows\Microsoft.NET\Framework\v3.5");
}
小点 - 但它可能既不是你自己的,也没有微软... – 2009-06-07 21:53:02