2015-05-18 16 views
9

Czy mogę wykryć (używając roslyn), że referencja x w ciele lambda jest zamknięta w stosunku do zmiennej zewnętrznej x, a nie jakieś zmienne lokalne względem samej lambda?Jak wykrywać zamknięcia w kodzie z Roslyn?

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

'x' będzie pole pewnej klasy zamknięcia, jeśli jest to wolny/zmiennej niewoli. – leppie

+3

Spójrz na [RoslynClrHeapAllocationAnalyzer] (https://github.com/mjsabby/RoslynClrHeapAllocationAnalyzer). Analizuje przydziały pamięci, a także zamknięcia. –

+1

@YuvalItzchakov dzięki za link, teraz jasne, że muszę sprawdzić właściwość 'Captured' obiektu' DataFlowAnalysis', który mogę pobrać z 'SemanticModel'. – Seldon

Odpowiedz

4

Yup. Możesz użyć interfejsu 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 
}