È noto in .NET che i tipi non sono garbage collection, il che significa che se stai giocando con f.ex. Reflection.Emit, devi stare attento a scaricare AppDomains e così via ... Almeno così ho capito come funzionano le cose.I tipi di dati MakeGenericType/generici vengono raccolti?
Questo mi ha fatto chiedo se i tipi generici sono garbage collection, per essere più precisi: i generici creati con MakeGenericType
, diciamo ... per esempio in base all'input dell'utente. :-)
Così ho costruito il seguente test case:
public interface IRecursiveClass
{
int Calculate();
}
public class RecursiveClass1<T> : IRecursiveClass
where T : IRecursiveClass,new()
{
public int Calculate()
{
return new T().Calculate() + 1;
}
}
public class RecursiveClass2<T> : IRecursiveClass
where T : IRecursiveClass,new()
{
public int Calculate()
{
return new T().Calculate() + 2;
}
}
public class TailClass : IRecursiveClass
{
public int Calculate()
{
return 0;
}
}
class RecursiveGenericsTest
{
public static int CalculateFromUserInput(string str)
{
Type tail = typeof(TailClass);
foreach (char c in str)
{
if (c == 0)
{
tail = typeof(RecursiveClass1<>).MakeGenericType(tail);
}
else
{
tail = typeof(RecursiveClass2<>).MakeGenericType(tail);
}
}
IRecursiveClass cl = (IRecursiveClass)Activator.CreateInstance(tail);
return cl.Calculate();
}
static long MemoryUsage
{
get
{
GC.Collect(GC.MaxGeneration);
GC.WaitForFullGCComplete();
return GC.GetTotalMemory(true);
}
}
static void Main(string[] args)
{
long start = MemoryUsage;
int total = 0;
for (int i = 0; i < 1000000; ++i)
{
StringBuilder sb = new StringBuilder();
int j = i;
for (int k = 0; k < 20; ++k) // fix the recursion depth
{
if ((j & 1) == 1)
{
sb.Append('1');
}
else
{
sb.Append('0');
}
j >>= 1;
}
total += CalculateFromUserInput(sb.ToString());
if ((i % 10000) == 0)
{
Console.WriteLine("Current memory usage @ {0}: {1}",
i, MemoryUsage - start);
}
}
Console.WriteLine("Done and the total is {0}", total);
Console.WriteLine("Current memory usage: {0}", MemoryUsage - start);
Console.ReadLine();
}
}
Come si può vedere, i tipi generici sono definiti 'forse ricorsivo', con una classe di 'coda' che segna la fine della ricorsione . E per garantire che GC.TotalMemoryUsage
non stia ingannando, ho anche aperto Task Manager.
Fin qui tutto bene. La prossima cosa che ho fatto è stato sparare questa bestia e mentre stavo aspettando un 'Out of memory' ... ho notato che era - contrariamente alle mie aspettative - non consumare più memoria nel tempo. In effetti, mostra un leggero calo nel consumo di memoria nel tempo.
Qualcuno può spiegarlo? I tipi generici sono effettivamente raccolti dal GC? E se è così ... ci sono anche i casi Reflection.Emit che sono raccolti?
Si potrebbe sperimentare la creazione di un 'WeakReference' al vostro tipo generico creato, quindi verificare se il' Target' è nullo dopo l'esecuzione di un passaggio GC. –
Forse questo aiuterà [Come i generici vengono compilati dal compilatore JIT?] (Http://stackoverflow.com/questions/5342345/how-do-generics-get-compiled-by-the-jit-compiler) –