web-dev-qa-db-ja.com

リフレクションを介して名前空間のすべての型を取得する

C#のリフレクションを使用して、名前空間のすべてのクラスを取得するにはどうすればよいですか

243
Chethan

次のコードは、現在のアセンブリで定義されている指定されたnamespaceのクラスの名前を出力します。
他の人が指摘したように、名前空間は異なるモジュール間に散在する可能性があるため、最初にアセンブリのリストを取得する必要があります。

string nspace = "...";

var q = from t in Assembly.GetExecutingAssembly().GetTypes()
        where t.IsClass && t.Namespace == nspace
        select t;
q.ToList().ForEach(t => Console.WriteLine(t.Name));
295
aku

FlySwatが言うように、複数のアセンブリにまたがる同じ名前空間を持つことができます(たとえばSystem.Collections.Generic)。まだロードされていない場合は、それらのアセンブリをすべてロードする必要があります。完全な答えを得るには:

AppDomain.CurrentDomain.GetAssemblies()
                       .SelectMany(t => t.GetTypes())
                       .Where(t => t.IsClass && t.Namespace == @namespace)

他のドメインのクラスが必要な場合を除き、これは機能するはずです。すべてのドメインのリストを取得するには、 このリンク に従ってください

69
nawfal
using System.Reflection;
using System.Collections.Generic;
//...

static List<string> GetClasses(string nameSpace)
{
    Assembly asm = Assembly.GetExecutingAssembly();

    List<string> namespacelist = new List<string>();
    List<string> classlist = new List<string>();

    foreach (Type type in asm.GetTypes())
    {
        if (type.Namespace == nameSpace)
            namespacelist.Add(type.Name);
    }

    foreach (string classname in namespacelist)
        classlist.Add(classname);

    return classlist;
}

注意:上記のコードは何が起こっているかを示しています。実装する場合は、簡易バージョンを使用できます。

using System.Linq;
using System.Reflection;
using System.Collections.Generic;
//...

static IEnumerable<string> GetClasses(string nameSpace)
{
    Assembly asm = Assembly.GetExecutingAssembly();
    return asm.GetTypes()
        .Where(type => type.Namespace == nameSpace)
        .Select(type => type.Name);
}
21
Ryan Farley

特定のアセンブリ、NameSpace、およびClassNameの場合:

var assemblyName = "Some.Assembly.Name"
var nameSpace = "Some.Namespace.Name";
var className = "ClassNameFilter";

var asm = Assembly.Load(assemblyName);
var classes = asm.GetTypes().Where(p =>
     p.Namespace == nameSpace &&
     p.Name.Contains(className) 
).ToList();

注:プロジェクトはアセンブリを参照する必要があります

15
John Peters

以下は、あるタイプが別のアセンブリのタイプをサブクラス化した場合に見つかる可能性が高いLoaderExceptionエラーの修正です。

// Setup event handler to resolve assemblies
AppDomain.CurrentDomain.ReflectionOnlyAssemblyResolve += new ResolveEventHandler(CurrentDomain_ReflectionOnlyAssemblyResolve);

Assembly a = System.Reflection.Assembly.ReflectionOnlyLoadFrom(filename);
a.GetTypes();
// process types here

// method later in the class:
static Assembly CurrentDomain_ReflectionOnlyAssemblyResolve(object sender, ResolveEventArgs args)
{
    return System.Reflection.Assembly.ReflectionOnlyLoad(args.Name);
}

これは、他のアセンブリで定義された型の読み込みに役立つはずです。

お役に立てば幸いです!

12
tsimon

名前空間は複数のアセンブリをブリッジできるため、名前空間内のすべての型を取得することはできませんが、アセンブリ内のすべてのクラスを取得し、それらがその名前空間に属しているかどうかを確認できます。

Assembly.GetTypes()はローカルアセンブリで機能します。または、最初にアセンブリをロードしてからGetTypes()を呼び出すことができます。

9
FlySwat

@aku answerと同じですが、拡張メソッドを使用します。

string @namespace = "...";

var types = Assembly.GetExecutingAssembly().GetTypes()
    .Where(t => t.IsClass && t.Namespace == @namespace)
    .ToList();

types.ForEach(t => Console.WriteLine(t.Name));
5
JoanComasFdz

1行だけの名前空間名の一部ですべてのクラスを取得します。

var allClasses = Assembly.GetExecutingAssembly().GetTypes().Where(a => a.IsClass && a.Namespace != null && a.Namespace.Contains(@"..your namespace...")).ToList();
5
Ivo Stoyanov

名前空間は、実際にはランタイムの設計においてかなり受動的であり、主に組織的なツールとして機能します。 .NETの型のフルネームは、名前空間とクラス/列挙/ Etcで構成されます。組み合わせた。特定のアセンブリのみを使用する場合は、アセンブリによって返される型をループするだけです。 GetExportedTypes() 型の値を確認します。 名前空間 現在のAppDomainにロードされたすべてのアセンブリを処理しようとした場合、AppDomain.CurrentDomainを使用する必要があります。 GetAssemblies()

3
TheXenocide
//a simple combined code snippet 

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Reflection;

namespace MustHaveAttributes
{
  class Program
  {
    static void Main ( string[] args )
    {
      Console.WriteLine ( " START " );

      // what is in the Assembly
      Assembly a = Assembly.Load ( "MustHaveAttributes" );
      Type[] types = a.GetTypes ();
      foreach (Type t in types)
      {

        Console.WriteLine ( "Type is {0}", t );
      }
      Console.WriteLine (
         "{0} types found", types.Length );

      #region Linq
      //#region Action


      //string @namespace = "MustHaveAttributes";

      //var q = from t in Assembly.GetExecutingAssembly ().GetTypes ()
      //        where t.IsClass && t.Namespace == @namespace
      //        select t;
      //q.ToList ().ForEach ( t => Console.WriteLine ( t.Name ) );


      //#endregion Action  
      #endregion

      Console.ReadLine ();
      Console.WriteLine ( " HIT A KEY TO EXIT " );
      Console.WriteLine ( " END " );
    }
  } //eof Program


  class ClassOne
  {

  } //eof class 

  class ClassTwo
  {

  } //eof class


  [System.AttributeUsage ( System.AttributeTargets.Class |
    System.AttributeTargets.Struct, AllowMultiple = true )]
  public class AttributeClass : System.Attribute
  {

    public string MustHaveDescription { get; set; }
    public string MusHaveVersion { get; set; }


    public AttributeClass ( string mustHaveDescription, string mustHaveVersion )
    {
      MustHaveDescription = mustHaveDescription;
      MusHaveVersion = mustHaveVersion;
    }

  } //eof class 

} //eof namespace 
2
Yordan Georgiev

とてもシンプル

Type[] types = Assembly.Load(new AssemblyName("mynamespace.folder")).GetTypes();
foreach (var item in types)
{
}
2
Antonio Lopes