2015-05-18 9 views
9

Posso rilevare (usando roslyn) che il riferimento x nel corpo lambda è la chiusura sulla variabile esterna x, non qualche variabile locale su lambda stessa?Come rilevare le chiusure nel codice con Roslyn?

var x = "foo"; 
var a = string[0]; 
a.Any(i => i == x); 
+3

'x' sarebbe un campo di una classe di chiusura, se è una variabile libera/catturato. – leppie

+3

Dai un'occhiata a [RoslynClrHeapAllocationAnalyzer] (https://github.com/mjsabby/RoslynClrHeapAllocationAnalyzer). Analizza allocazioni di memoria e chiusure. –

+1

@YuvalItzchakov grazie per il collegamento, ora è chiaro che ho bisogno di controllare la proprietà 'Captured' dell'oggetto' DataFlowAnalysis' che posso ottenere da 'SemanticModel'. – Seldon

risposta

4

Yup. Puoi utilizzare l'API DataFlowAnalysis.

var tree = CSharpSyntaxTree.ParseText(
    @" 
class C{ 
void M(){ 
    var x = ""foo""; 
    var a = new string[0]; 
    var testing = a.Any(i => i == x); 
} 
} 
"); 
var Mscorlib = PortableExecutableReference.CreateFromAssembly(typeof(object).Assembly); 
var compilation = CSharpCompilation.Create("MyCompilation", 
    syntaxTrees: new[] { tree }, references: new[] { Mscorlib }); 
var model = compilation.GetSemanticModel(tree); 

var lambda = tree.GetRoot().DescendantNodes().OfType<LocalDeclarationStatementSyntax>().Last(); 

var dataFlowAnalysis = model.AnalyzeDataFlow(lambda); 
var capturedVariables = dataFlowAnalysis.Captured; 

foreach(var variable in capturedVariables) 
{ 
    //Do something 
}