This really has my stumped today. I'm sure its simple, but... Here is my sample code:
using System;
using System.Collections;
namespace ConsoleApplication1
{
class Program
{
public ArrayList SomeProp { get; set; }
static void Main(string[] args)
{
// Get the Type of a property by reflection.
Type myPropType = typeof(Program).GetProperty("SomeProp").PropertyType;
// Create an instance of the determined Type.
var x = Activator.CreateInstance(myPropType);
// Now try to use that instance, passing to a method that takes a generic.
WhatAmI(x);
Console.ReadKey();
}
private static void WhatAmI<T>(T x)
{
Console.WriteLine("T is: " + typeof(T).FullName);
Console.WriteLine("x is: " + x.GetType().FullName);
}
}
}
The output here is:
T is: System.Object
x is: System.Collections.ArrayList
Basically what is going on here is I have some property on some class. It doesnt really matter what/where that comes from. The important part is that I get the PropertyInfo for that property. From there I create a new instance of its Type with Activator.
Now If I pass that created instance to a function that takes a generic parameter, the generic Type always comes across as Object, because Activator.CreateInstance() returns an Object, so "var x" is an Object, not, in this case, an ArrayList.
What I need to happen is for the generic type to be the actual type, int his case "ArrayList".
I know there is an Activator.CreateInstance() that I can use instead, but I can't use a Type (PropertyInfo.PropertyType) within the angle brackets for that method.
I could also just cast the returned object, like:
myPropType x = (myPropType)Activator.CreateInstance(myPropType);
But obviously that doesn't compile... Plus it wouldn't be valid anyway because the cast is compile time, and I don't know the type until runtime, but conceptually its what I need...
So I'm stuck with this Type, but I can't figure out how to get it passed over to the WhatAmI() method, and have T be ArrayList, not Object.
Ideas?
-
To call generic methods using a
Type
at runtime, you need to use reflection - andMakeGenericMethod
in particular, something like:typeof(Program).GetMethod("WhatAmI", BindingFlags.Static | BindingFlags.NonPublic) .MakeGenericMethod(x.GetType()).Invoke(null, new object[] { x });
Otherwise, the compiler infers the
<T>
from the variable type -object
in this case.One side point here is that you can improve things by only doing the reflection once - i.e. don't use
Activator
, but use a generic constraint instead:private static void WhatAmI<T>() where T : new() { T x = new T(); Console.WriteLine("T is: " + typeof(T).FullName); Console.WriteLine("x is: " + x.GetType().FullName); }
Defines the method without an arg, but with a default constructor; then:
// Get the Type of a property by reflection. Type myPropType = typeof(Program).GetProperty("SomeProp").PropertyType; // Now call a generic method just using the type typeof(Program).GetMethod("WhatAmI", BindingFlags.Static | BindingFlags.NonPublic) .MakeGenericMethod(myPropType).Invoke(null, null);
calls the method just using the type - the instance is created inside the method. This is usually faster than repeated reflection.
rally25rs : Fortuantely the place I need to apply this, it is my own method that I know is generic, so this should work. I could see this becoming a real issue for shared libraries, where you may inadvertantly call a method that you don't know is generic. Anyway, I'll give this a shot. Thanks!rally25rs : I've never seen that "where T : new()" before. Very cool! Thanks!
0 comments:
Post a Comment