below is a code example and the question, please note that I can NOT use C# 4.0 and the dynamic keyword.
static class TestClass
{
static void Main(string[] args)
{
Object o = "Previous value";
Test(ref o);
Trace.WriteLine(o);
}
static public void Test<T>(ref T obj)
{
// The goal is to somehow invoke Test2 with the real type of obj, i.e the type in obj.GetType()
// 1st try:
Test2(ref obj); // This doesn't work because the type in Test2 will be the same as T here.
// 2nd try:
MethodInfo mi = typeof(TestClass).GetMethod("Test2");
mi = mi.MakeGenericMethod(new Type[] { obj.GetType() });
mi.Invoke(null, new Object[] { obj }); // obj is no longer by reference so we need to store the object array and copy back the result after the call
// 3rd try, successful implementation by the smartest mind of stack overflow :)
}
static public void Test2<T>(ref T s)
{
if (typeof(T) == typeof(String))
{
s = (T)(Object)"Hello world!";
}
}
}
I also tried some more methods using Delegate.CreateDelegate but without any luck. Is this at all possible?
Edit: I'm not afraid of using a dynamic method (and MSIL assembler) but my knowledge in this area is very limited.
Edit2: Here's an example that is closer to what I really trying to do:
public static class TypeHandler<T>
{
public delegate void ProcessDelegate(ref T value);
public static readonly ProcessDelegate Process = Init();
private static ProcessDelegate Init()
{
// Do lot's of magic stuff and returns a suitable delegate depending on the type
return null;
}
}
static class TestClass
{
static public void Main(string[] args)
{
Object o = "Previous value";
Test(ref o);
Trace.WriteLine(o);
}
static public void Test<T>(ref T obj)
{
if (obj is T)
{
// Optimized, common case
TypeHandler<T>.Process(ref obj);
return;
}
Type t = obj.GetType();
// How to call the delegate found in TypeHandler<t>.Process ? (I can get delegate but I can't call it).
}
}
Update 3: OK, since you're fine with an ugly solution, you may want to check out the undocumented __refvalue
and __makeref
keywords.
It seems your issue is that you want to be able to specify the type for a ref object
parameter to be converted or changed to.
The problem with that is that you can't just arbitrarily assign a variable of any type T
to a string
, for example. So you'd need to pass in a ref object
or ref string
for that to work at all.
It looks to me like oberfreak nailed down what you were trying to achieve (I originally failed to notice you had initialized o
as a string
and thus clearly wanted its actual type to influence the behavior of the Test2
function). His answer has the right approach for you.
Update: you mention in a comment that what you're trying to do is have dynamic behavior which could be achieved using a dictionary. I'm guessing that looks something like this?
Update 2: updated this example based on your updated example.
public static class TypeHandler // note: get rid of generic T parameter
{
delegate void ProcessDelegate(ref object obj); // again, not generic
static Dictionary<Type, ProcessDelegate> processors = new Dictionary<Type, ProcessDelegate>()
{
{ typeof(string), (ref object obj) => { obj = "Hello, world!"; } }
// etc.
};
public static void Process(ref object obj)
{
processors[obj.GetType()].Invoke(ref obj);
}
}
That should work. But you can't really achieve the same thing with generics because there's no way to do this (as you know):
// not allowed
// |
// -----------
// | |
TypeHandler<o.GetType()>.Process(ref o);
If there were, then you'd be all set. But the only possible way you can do that is by using reflection, which is ugly and expensive for something like this and would clearly defeat your intention of keeping this simple and ensuring good performance.