本文告诉大家如何在使用 IIncrementalGenerator 进行增量的 Source Generator 生成代码时,如何获取到当前正在分析的程序集所引用的所有的程序集,以及引用的程序集里面的所有类型

这项技术可以用在生成导出类型相关的需求上,比如我想导出我当前程序集里面所有引用的程序集的继承于 IFoo 接口的所有类型,即可采用本文介绍的方法

核心逻辑是在 Compilation 里面拿到 SourceModule 属性,通过 IModuleSymbol 类型的 SourceModule 属性进一步拿到 ImmutableArray<IAssemblySymbol> 类型的 ReferencedAssemblySymbols 属性

这里的 ReferencedAssemblySymbols 属性就是当前的程序集所引用的程序集了

在这些程序集上枚举所有程序集内的语义类型即可获取到所有的类型

以下是详细的例子

为了方便描述本文的技术实现,需要创建三个项目,分别是 App 和 Lib 和 Analyzers 三个项目。在本文末尾将可以找到所有代码的下载方法

这里 App 项目是用来被分析器项目 Analyzers 项目进行分析的。而 Lib 项目则是一个基础库,被 App 项目所引用

在这个例子里面,咱的任务就是在 Analyzers 分析器项目里面编写代码,分析去 App 里面所引用的 Lib 项目里面包含的所有类型

具体的初始化方法就是新建三个 .NET 7 控制台项目,分别命名为 App 和 Lib 和 Analyzers 项目。接着编辑 Lib 项目修改为类库项目,修改的方法就是编辑 csproj 项目文件,替换为如下代码

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup> </Project>

接着修改 App 项目的 csproj 项目文件,让 App 项目引用 Lib 项目以及 Analyzers 分析器项目。只有让 App 项目引用 Analyzers 分析器项目,才可以让 Analyzers 分析器项目对 App 项目进行分析,编辑之后的 csproj 项目文件代码如下

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup> <ItemGroup>
<ProjectReference Include="..\Lib\Lib.csproj" />
<ProjectReference Include="..\Analyzers\Analyzers.csproj" OutputItemType="Analyzer" ReferenceOutputAssembly="false" />
</ItemGroup> </Project>

如以上代码,需要让 App 项目引用 Analyzers 分析器项目,设置引用 OutputItemType 为 Analyzer 才可以让 Analyzers 项目被当成 App 项目的分析器

由于 App 项目不需要用到任何在 Analyzers 分析器项目定义的类型,于是也设置了 ReferenceOutputAssembly 为 false 值。通过 OutputItemType="Analyzer" ReferenceOutputAssembly="false" 两个属性即可让 Analyzers 项目只作为 App 项目的分析器存在,不影响 App 项目的其他逻辑,也不会让 App 项目真正引用到 Analyzers 项目里面的任何公开类型

同时设置了 App 项目引用 Analyzers 分析器项目,即可在构建的时候,先构建 Analyzers 分析器项目,再构建 App 项目,确定了项目的构建顺序。于是在 Analyzers 分析器项目里面编写的 IIncrementalGenerator 增量 Source Generator 生成代码逻辑将可以被正常执行

最后来到最重要的 Analyzers 分析器项目。为了能够让 VisualStudio 开森以及让 dotnet 开心,推荐使用的是 netstandard2.0 框架。然后引用上必要的 NuGet 包,修改之后的 csproj 项目文件代码如下

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
<TargetFramework>netstandard2.0</TargetFramework> <EnforceExtendedAnalyzerRules>true</EnforceExtendedAnalyzerRules>
<LangVersion>latest</LangVersion> <Nullable>enable</Nullable>
</PropertyGroup> <ItemGroup>
<PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.2.0" PrivateAssets="all" />
</ItemGroup> </Project>

详细关于以上 csproj 项目文件代码里的 EnforceExtendedAnalyzerRules 的属性,请参阅 Roslyn 分析器 EnforceExtendedAnalyzerRules 属性的作用

以上的 LangVersion 属性设置为 latest 表示使用最新的语言版本,详细请参阅 VisualStudio 使用三个方法启动最新 C# 功能

通过以上配置即可完成项目的初始化逻辑。回到咱这个例子的任务上,就是在 Analyzers 分析器项目编写代码,分析 App 项目所引用的 Lib 项目里面的存在哪些类型

为了能够让 Analyzers 分析器项目有活干,咱就来给 Lib 项目多添加一些随意的类型

namespace Lib;

public class Base
{
} public class Foo1 : Base
{
} public class Foo2 : IFoo
{
} public class Foo3 : Base, IFoo
{
} public interface IFoo
{
}

完成准备工作之后,接下来开始本文的核心逻辑编写。先在 Analyzers 分析器项目上新建一个继承 IIncrementalGenerator 接口的 FooTelescopeIncrementalGenerator 类型,接下来的核心逻辑将在 FooTelescopeIncrementalGenerator 的 Initialize 开始编写

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; using Microsoft.CodeAnalysis; namespace Analyzers; [Generator(LanguageNames.CSharp)]
public class FooTelescopeIncrementalGenerator : IIncrementalGenerator
{
public void Initialize(IncrementalGeneratorInitializationContext context)
{
... // 忽略代码
}
}

根据上文的描述,咱需要先从 context 里面的 CompilationProvider 获取到引用的程序集,代码如下

[Generator(LanguageNames.CSharp)]
public class FooTelescopeIncrementalGenerator : IIncrementalGenerator
{
public void Initialize(IncrementalGeneratorInitializationContext context)
{
var typeNameIncrementalValueProvider = context.CompilationProvider.Select((compilation, token) =>
{
... // 忽略代码
});
}
}

通过 compilation 的 SourceModule 属性的 ReferencedAssemblySymbols 即可获取到所有的引用程序集,如以下代码

[Generator(LanguageNames.CSharp)]
public class FooTelescopeIncrementalGenerator : IIncrementalGenerator
{
public void Initialize(IncrementalGeneratorInitializationContext context)
{
var typeNameIncrementalValueProvider = context.CompilationProvider.Select((compilation, token) =>
{
// 获取到所有引用程序集
var referencedAssemblySymbols = compilation.SourceModule.ReferencedAssemblySymbols; ... // 忽略代码
});
}
}

遍历所有引用的程序集。这里为了博客方便,就只获取 Lib 程序集里面的类型

    public void Initialize(IncrementalGeneratorInitializationContext context)
{
var typeNameIncrementalValueProvider = context.CompilationProvider.Select((compilation, token) =>
{
// 获取到所有引用程序集
var referencedAssemblySymbols = compilation.SourceModule.ReferencedAssemblySymbols; // 为了方便代码理解,这里只取名为 Lib 程序集的内容…
foreach (var referencedAssemblySymbol in referencedAssemblySymbols)
{
var name = referencedAssemblySymbol.Name; if (name.Contains("Lib"))
{
... // 在这里编写获取程序集所有类型的代码
}
else
{
// 其他的引用程序集,在这里就忽略
}
}
... // 忽略代码
});
}

通过 IAssemblySymbol 的 GlobalNamespace 属性即可获取到顶层的 INamespaceSymbol 符号,通过语义知识可以了解到,类型都是存放在命名空间里面的,只需要对命名空间进行递归即可获取到所有的类型

如以下代码即可递归获取某个 INamespaceSymbol 下的所有类型

    private static IEnumerable<INamedTypeSymbol> GetAllTypeSymbol(INamespaceSymbol namespaceSymbol)
{
var typeMemberList = namespaceSymbol.GetTypeMembers(); foreach (var typeSymbol in typeMemberList)
{
yield return typeSymbol;
} foreach (var namespaceMember in namespaceSymbol.GetNamespaceMembers())
{
foreach (var typeSymbol in GetAllTypeSymbol(namespaceMember))
{
yield return typeSymbol;
}
}
}

给 GetAllTypeSymbol 传入程序集 GlobalNamespace 即可获取到此程序集里面的所有类型,代码如下

    public void Initialize(IncrementalGeneratorInitializationContext context)
{
var typeNameIncrementalValueProvider = context.CompilationProvider.Select((compilation, token) =>
{
// 获取到所有引用程序集
var referencedAssemblySymbols = compilation.SourceModule.ReferencedAssemblySymbols; // 为了方便代码理解,这里只取名为 Lib 程序集的内容…
foreach (var referencedAssemblySymbol in referencedAssemblySymbols)
{
var name = referencedAssemblySymbol.Name; if (name.Contains("Lib"))
{
// 获取所有的类型
// 这里 ToList 只是为了方便调试
var allTypeSymbol = GetAllTypeSymbol(referencedAssemblySymbol.GlobalNamespace).ToList();
}
else
{
// 其他的引用程序集,在这里就忽略
}
}
... // 忽略代码
});
}

以上拿到的 allTypeSymbol 就是引用的 Lib 程序集里面的所有类型。为了测试咱的分析器代码是否正确,可以尝试将收集到的 Lib 程序集里面的所有类型的记录输出作为一个源代码生成

{% raw %}

    public void Initialize(IncrementalGeneratorInitializationContext context)
{
Debugger.Launch(); var typeNameIncrementalValueProvider = context.CompilationProvider.Select((compilation, token) =>
{
var typeNameList = new List<string>(); // 获取到所有引用程序集
var referencedAssemblySymbols = compilation.SourceModule.ReferencedAssemblySymbols; // 为了方便代码理解,这里只取名为 Lib 程序集的内容…
foreach (IAssemblySymbol? referencedAssemblySymbol in referencedAssemblySymbols)
{
var name = referencedAssemblySymbol.Name; if (name.Contains("Lib"))
{
// 获取所有的类型
// 这里 ToList 只是为了方便调试
var allTypeSymbol = GetAllTypeSymbol(referencedAssemblySymbol.GlobalNamespace).ToList(); foreach (var typeSymbol in allTypeSymbol)
{
typeNameList.Add(typeSymbol.ToDisplayString());
}
}
else
{
// 其他的引用程序集,在这里就忽略
}
} return typeNameList;
}); context.RegisterSourceOutput(typeNameIncrementalValueProvider, (productionContext, list) =>
{
var code = $@"
public static class FooHelper
{{
public static IEnumerable<string> GetAllTypeName()
{{
{(string.Join("\r\n", list.Select(t => $@"yield return ""{t}"";")))}
}}
}}";
productionContext.AddSource("FooHelper", code);
});
}

{% endraw %}

如以上代码就在代码生成器里面生成了名为 FooHelper 的类型,这个类型将会返回 Lib 程序集里面的所有的类型

接下来编辑 App 项目的 Program.cs 文件,替换为如下代码

foreach (var name in FooHelper.GetAllTypeName())
{
Console.WriteLine(name);
}

假设分析器项目代码编写正确,那就可以成功输出 Lib 程序集里面的所有类型到控制台

试试运行一下项目,看看写的对不对吧

本文所有代码放在 githubgitee 上,可以通过以下方式获取整个项目的代码

先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin d0b5dc0af9c9f4ff3c18a2212200b492e3edbc08

以上使用的是 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin d0b5dc0af9c9f4ff3c18a2212200b492e3edbc08

获取代码之后,进入 LurwajedalwemcejemQallneleecairwhair 文件夹

IIncrementalGenerator 获取引用程序集的所有类型的更多相关文章

  1. 使用typeid(变量或类型).name()来获取常量或变量的类型---gyy整理

    使用typeid(变量或类型).name()来获取常量或变量的类型 <typeinfo>  该头文件包含运行时类型识别(在执行时确定数据类型)的类 typeid的使用   typeid操作 ...

  2. Ⅴ.spring的点点滴滴--引用其他对象或类型的成员

    承接上文 引用其他对象或类型的成员 .net篇(环境为vs2012+Spring.Core.dll v1.31) public class Person { public string Name { ...

  3. C# -- 使用反射(Reflect)获取dll文件中的类型并调用方法

    使用反射(Reflect)获取dll文件中的类型并调用方法 需引用:System.Reflection; 1. 使用反射(Reflect)获取dll文件中的类型并调用方法(入门案例) static v ...

  4. XML序列化 判断是否是手机 字符操作普通帮助类 验证数据帮助类 IO帮助类 c# Lambda操作类封装 C# -- 使用反射(Reflect)获取dll文件中的类型并调用方法 C# -- 文件的压缩与解压(GZipStream)

    XML序列化   #region 序列化 /// <summary> /// XML序列化 /// </summary> /// <param name="ob ...

  5. C#开发BIMFACE系列9 服务端API之获取应用支持的文件类型

    系列目录     [已更新最新开发文章,点击查看详细] BIMFACE最核心能力之一是工程文件格式转换.无需安装插件,支持数十种工程文件格式在云端转换,完整保留原始文件信息.开发者将告别原始文件解析烦 ...

  6. Java web项目引用java项目,类型找不到

    Java web项目引用java项目,类型找不到 错误信息: java.lang.ClassNotFoundException: org.codehaus.jackson.map.ObjectMapp ...

  7. java利用反射获取类的属性及类型

    java利用反射获取类的属性及类型. import java.lang.reflect.Field; import java.math.BigDecimal; import java.util.Map ...

  8. 获取jdk支持的编码类型

    //获取jdk支持的编码类型 Map<String,Charset> maps = Charset.availableCharsets(); for(Map.Entry<String ...

  9. VS2017 加载项目 :未找到框架“.NETFramework,Version=v4.7”的引用程序集(出坑指南)

    报出的错误为: 错误MSB3644: 未找到框架“.NETFramework,Version=v4.7”的引用程序集.若要解决此问题,请安装此框架版本的 SDK 或 Targeting Pack,或将 ...

  10. Perl 引用:引用就是指针,Perl 引用是一个标量类型可以指向变量、数组、哈希表(也叫关联数组)甚至子程序。

    Perl 引用引用就是指针,Perl 引用是一个标量类型可以指向变量.数组.哈希表(也叫关联数组)甚至子程序,可以应用在程序的任何地方. 1.创建引用1.使用斜线\定义变量的时候,在变量名前面加个\, ...

随机推荐

  1. FPGA模块化设计

    模块化设计出发点 在实际地操作中,总有一些基础的模块需要不断地寻找,往往需要消耗大量的时间.为了节约模块化设计的时间,提高设计的效率.在这里将一些基础的模块全部进行封装,利用网络的便捷性,实现快速地基 ...

  2. KingbaseES V8R3集群运维案例---进程内核参数配置集群启动故障

    案例说明: KingbaseES V8R3集群在部署时需要配置与进程间通讯(IPC)相关的内核参数,如果缺失配置或配置错误,kingbasecluster服务在启动过程中将因为内核参数配置错误导致启动 ...

  3. kingbaseES V8R3集群运维案例之---集群部署前后ssh端口修改

    kingbaseES V8R3集群运维案例之---集群部署前后ssh端口修改 案例说明: kingbaseES V8R3集群部署读写分离的集群是使用ssh的默认端口(22)部署,当改为非默认端口时,在 ...

  4. KingbaseES V8R6集群运维系列 -- 修改ssh通信为 sys_securecmdd 通信

    一.适用于: 本文档使用于KingbaseES V008R006版本. 二.关于SYS_SECURECMDD: sys_securecmdd是KingbaseES集群自带的工具,集群监控.管理集群时通 ...

  5. 【已解决】初始化 Hive 元数据库报错slf4j-log4j12-1.7.25.jar包冲突

    错误log描述 [root@hadoop102 hive]# schematool -initSchema -dbType mysql -verboseSLF4J: Class path contai ...

  6. #dp#洛谷 3244 [HNOI2015]落忆枫音

    题目 分析 每个有入度的点可以选择任意一个父节点组成一棵树,那么原来的答案就是 \(\prod_{i=2}^ndeg[i]\) 现在多了一条边,如果边的终点是1或者它是一个自环那么可以不用管这条边. ...

  7. #背包#AT2037 [ARC060A] 高橋君とカード / Tak and Cards

    题目 有一个长度为\(n\)的数组\(a\),选择若干个数使它们的平均数为\(A\),问共有多少种方案 分析 设\(dp[i][j]\)表示选择\(i\)个数总和为\(j\)的方案数,那么答案就是\( ...

  8. 数据库操作入门:PyMongo 和 MongoDB 的基本用法

    MongoDB MongoDB是一种流行的NoSQL数据库,它将数据存储在类似JSON的文档中,使数据库非常灵活和可扩展 PyMongo Python需要一个MongoDB驱动程序来访问MongoDB ...

  9. Prometheus之node_exporter安装

    一.简介 node_exporter用来安装到被监控的主机上,暴露被监控主机的指标数据,服务器端基于http协议调用的端口9100(默认)来获取被监控服务器信息. 二.安装部署 下载地址 https: ...

  10. redis 简单整理——HyperLogLog[十三]

    前言 简单介绍一下HyperLogLog. 正文 HyperLogLog并不是一种新的数据结构(实际类型为字符串类型),而 是一种基数算法,通过HyperLogLog可以利用极小的内存空间完成独立总数 ...