【问题标题】:Roslyn Find ALL References to a Method in ALL Projects in a SolutionRoslyn 在解决方案的所有项目中查找对方法的所有引用
【发布时间】:2020-06-24 14:22:14
【问题描述】:

我一直在尝试在整个解决方案中找到对方法的所有引用。有人会认为 SymbolFinder.FindReferencesAsync 将 ISymbol 作为参数进行搜索,而解决方案对象会这样做,但它不会......

所有文章和 SO 问题/答案似乎都将其作为解决方案。它适用于在包含它的程序集中查找对方法的所有引用,但不适用于引用它的任何项目。我有一个理论,这是因为当您进行引用时,引用程序集中的方法符号不同,他们希望您获取该符号,然后使用 SymbolFinder.FindSourceDeclarationsAsync 将其解析为您正在寻找的实际符号。

如果是这样的话,我真的不知道该怎么做?也许以某种方式获取程序集中所有符号的列表,然后对每个人进行 FindSource 调用并比较它以查看它是否引用了您的 MethodSymbol?我当然希望不会!

这是一个简单的例子

public void Find2()
{
    string solutionPath = @"C:\Projects\RoslynTestTargetProject\RoslynTestTargetProject.sln";
    var msWorkspace = MSBuildWorkspace.Create();
    var solution = msWorkspace.OpenSolutionAsync(solutionPath).Result;
    var methodProjectComp = solution.Projects.Single(x => x.Name == "RoslynTestTargetProject").GetCompilationAsync().Result;

    // I have loaded these all kinds of different ways but it never impacts the end result
    var classWithMethodToSearchFor = methodProjectComp.GetTypeByMetadataName("RoslynTestTargetProject.Class1");
    var methodToSearchFor = classWithMethodToSearchFor.GetMembers("GetPerson").First() as IMethodSymbol;

    var references = SymbolFinder.FindReferencesAsync(methodToSearchFor, solution).Result;
    var callers = SymbolFinder.FindCallersAsync(methodToSearchFor, solution).Result;

    int c = references.Select(x => x.Locations.Count()).Sum(); 
}

这是具有我正在搜索的方法和一个引用的类。

public class Class1
{
    public Class1() {}        
    public Person GetPerson(Guid id) => new Person();
    public IEnumerable<Person> GetPeople(IEnumerable<Guid> ids) => ids.Select(x => GetPerson(x));
}

对它的另一个引用只是一个简单的控制台项目,具有以下内容:

public static void GetPersonOutsideReference()
{
    var c = new Class1();
    c.GetPerson(Guid.NewGuid());
} 

我希望 Find2 方法末尾的计数返回 2,类中的引用为 1,其他项目中的引用为 1,但它始终为 1。

我尝试了几种方法,包括其他 SO 帖子:123,但均未成功。

【问题讨论】:

  • 你有项目引用或DLL引用吗?
  • 您需要有一个源符号而不是引用符号才能进行查找所有引用。

标签: c# roslyn


【解决方案1】:

我想出了一种似乎是正确的方法。我还没有看到提供解决方案中所有参考的解决方案或答案。

您首先要在项目中查询方法所属的类型(类)。正如我在问题中提到的那样,即使多个项目可能引用和使用它们相同的类型,但它们没有该类型的相同符号。因此,您需要做的是通过传递 MetadataName 来询问每个编译是否具有匹配的类型。

foreach (var project in solution.Projects)
{
     var compilation = project.GetCompilationAsync().Result;

      // Look for a reference to the Class in the Assembly
      var classRef = compilation.GetTypeByMetadataName("RoslynTestTargetProject.Class1");

如果引用不为空,则表示该类型正在程序集中使用。现在您可以检查对特定于该编译的方法的符号的引用,如下所示:

//For Methods
var refs = SymbolFinder.FindReferencesAsync(mSymbol, solution).Result;

//For Constructors
var refs = classRef.Constructors.Select(m => SymbolFinder.FindReferencesAsync(m, solution).Result).ToList();

【讨论】:

    猜你喜欢
    • 2012-07-18
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-01-03
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多