How to get a Type from a C# type name string?

后端 未结 2 725
谎友^
谎友^ 2021-01-25 05:27

I\'ve been looking at return values for Type.Namespace, Type.Name, Type.FullName, and Type.AssemblyQualifiedName. There are

相关标签:
2条回答
  • 2021-01-25 06:06

    So, basically per our conversation in the comments the goal here is to get a type object (for whatever purpose) from a name of the type. Calling "Type.GetType()" works for simple types but for generics and other types it doesn't work because the name needs to be qualified. The key, then is to use the code compiler to actually have the C# code engine find and get the type. The following is a working program that does just that:

    using System;
    using System.Reflection;
    using Microsoft.CSharp;
    using System.CodeDom.Compiler;
    
    namespace SimpleCompileTest
    {
        class Program
        {
            public static void Main(string[] args)
            {
                string typeName = "System.Collections.Generic.List<int>";
                Type theType = GetTypeFromName(typeName);
            }
    
            private static Type GetTypeFromName(string typeName)
            {
                // double open and close are for escape purposes
                const string typeProgram = @"using System; using System.Collections.Generic; using System.IO;
                    namespace SimpleTest
                    {{
                        public class Program
                        {{
                            public static Type GetItemType()
                            {{
                                {0} typeTest = new {0}();
                                if (typeTest == null) return null;
                                return typeTest.GetType();
                            }}
                        }}
                    }}";
    
                var formattedCode = String.Format(typeProgram, typeName);
    
                var CompilerParams = new CompilerParameters
                    {
                        GenerateInMemory = true,
                        TreatWarningsAsErrors = false,
                        GenerateExecutable = false,
                        CompilerOptions = "/optimize"
                    };
    
                string[] references = { "System.dll" };
                CompilerParams.ReferencedAssemblies.AddRange(references);
    
                var provider = new CSharpCodeProvider();
                CompilerResults compile = provider.CompileAssemblyFromSource(CompilerParams, formattedCode);
                if (compile.Errors.HasErrors) return null;
    
    
                Module module = compile.CompiledAssembly.GetModules()[0];
                Type mt = null; MethodInfo methInfo = null;
    
                if (module != null) mt = module.GetType("SimpleTest.Program");
                if (mt != null) methInfo = mt.GetMethod("GetItemType");
                if (methInfo != null) return (Type)methInfo.Invoke(null, null);
    
                return null;
            }
        }
    }
    

    One really important thing to note - you need to add the list of assemblies to the compiler that you hope to pull types from. That means if you have a custom type that you want to reference you need to provide that to the compiler, but otherwise, this works! Enjoy!

    0 讨论(0)
  • 2021-01-25 06:23

    I've managed to achieve this now with a single line of code in each direction translating to and from friendly type names and runtime Type instances. Incredible. And some said it was not possible at all, lol. Flawless.

    static Type GetType( string friendlyName )
    {
        return (Type)(new CSharpCodeProvider().CompileAssemblyFromSource( new CompilerParameters( AppDomain.CurrentDomain.GetAssemblies().SelectMany<Assembly,string>( a => a.GetModules().Select<Module,string>( m => m.FullyQualifiedName )).ToArray(), null, false) {GenerateExecutable = false, GenerateInMemory = true, TreatWarningsAsErrors = false, CompilerOptions = "/optimize"}, "public static class C{public static System.Type M(){return typeof(" + friendlyName + ");}}").CompiledAssembly.GetExportedTypes()[0].GetMethod("M").Invoke( null, System.Reflection.BindingFlags.Static, null, null, null ));
    }
    
    static string GetFriendlyName( Type type )
    {
        return new CSharpCodeProvider().GetTypeOutput(new CodeTypeReference(type));
    }
    

    The above code (first method only), when expanded to multiple lines looks like the following. You can just make a call like GetType("System.Collections.Generic.List<int>"); and it will return a type reference.

    static Type GetType( string friendlyName )
    {
        var currentlyLoadedModuleNames = AppDomain.CurrentDomain.GetAssemblies().SelectMany<Assembly,string>( a => a.GetModules().Select<Module,string>( m => m.FullyQualifiedName )).ToArray();
        var csc = new CSharpCodeProvider();
        CompilerResults results = csc.CompileAssemblyFromSource(
            new CompilerParameters( currentlyLoadedModuleNames, "temp.dll", false) {
                GenerateExecutable = false, GenerateInMemory = true, TreatWarningsAsErrors = false, CompilerOptions = "/optimize"
            },
            @"public static class TypeInfo {
                public static System.Type GetEmbeddedType() {
                return typeof(" + friendlyName + @");
                }
            }");
        if (results.Errors.Count > 0)
            throw new Exception( "Error compiling type name." );
        Type[] type = results.CompiledAssembly.GetExportedTypes();
        return (Type)type[0].GetMethod("GetEmbeddedType").Invoke( null, System.Reflection.BindingFlags.Static, null, null, null );
    }
    

    Update: I added a line to make all the modules loaded in the current app domain available to the compiler. That should guarantee that this is able to acquire any type by name as if you had referenced it directly in your code, with the exception that the desired types have to be public, since they are essentially referenced from an external assembly from within the compiler rather than directly in the currently executing assembly.

    Just as as test, the returned type should work in a cache, since:

    Type t = GetType( "System.Collections.Generic.List<int>" );
    Console.WriteLine( typeof(System.Collections.Generic.List<int>) == t );
    //RETURNS TRUE
    
    0 讨论(0)
提交回复
热议问题