翼度科技»论坛 编程开发 .net 查看内容

.Net Core 你必须知道的source-generators

7

主题

7

帖子

21

积分

新手上路

Rank: 1

积分
21
源生成器是 C# 9 中引入的一项功能,允许在编译过程中动态生成代码
它们直接与 C# 编译器集成(Roslyn)并在编译时运行,分析源代码并根据分析结果生成附加代码
源生成器提供了一种简化的自动化代码生成方法,无需外部工具或单独的预编译步骤。
通过无缝集成到编译过程中,源生成器可以提高生产力、减少错误并实现更高效的开发工作流程。
 
如何使用
创建 .NET 控制台应用程序。 此示例使用 .NET 6。将 Program 类替换为以下代码。 
  1. namespace ConsoleApp;
  2. partial class Program
  3. {
  4.     static void Main(string[] args)
  5.     {
  6. <ItemGroup>
  7.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  8.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  9. </ItemGroup>HelloFrom("Generated Code");
  10.     }
  11.     static partial void HelloFrom(string name);
  12. }
复制代码
接下来,我们将创建一个源生成器项目来实现 partial void HelloFrom 方法对应项。
创建一个以 netstandard2.0 目标框架名字对象为目标的 .NET 类库。 添加以下 NuGet 包
  1. <ItemGroup>
  2.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  3.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  4. </ItemGroup>
复制代码
 
然后,创建一个名为 HelloSourceGenerator.cs 的新 C# 文件,该文件指定你自己的源生成器,如下所示:
  1. using Microsoft.CodeAnalysis;namespace SourceGeneratorInCSharp{    [Generator]    public class SourceGenerator : ISourceGenerator    {<ItemGroup>
  2.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  3.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  4. </ItemGroup>public void Execute(GeneratorExecutionContext context)<ItemGroup>
  5.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  6.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  7. </ItemGroup>{<ItemGroup>
  8.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  9.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  10. </ItemGroup>    // Find the main method<ItemGroup>
  11.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  12.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  13. </ItemGroup>    var mainMethod = context.Compilation.GetEntryPoint(context.CancellationToken);<ItemGroup>
  14.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  15.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  16. </ItemGroup>    // Build up the source code<ItemGroup>
  17.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  18.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  19. </ItemGroup>    string source = $@"// using System;namespace {mainMethod.ContainingNamespace.ToDisplayString()}{{    public static partial class {mainMethod.ContainingType.Name}    {{<ItemGroup>
  20.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  21.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  22. </ItemGroup>static partial void HelloFrom(string name) =><ItemGroup>
  23.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  24.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  25. </ItemGroup>    Console.WriteLine($""Generator says: Hi from '{{name}}'"");    }}}}";<ItemGroup>
  26.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  27.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  28. </ItemGroup>    var typeName = mainMethod.ContainingType.Name;<ItemGroup>
  29.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  30.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  31. </ItemGroup>    // Add the source code to the compilation<ItemGroup>
  32.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  33.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  34. </ItemGroup>    context.AddSource($"{typeName}.g.cs", source);<ItemGroup>
  35.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  36.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  37. </ItemGroup>}<ItemGroup>
  38.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  39.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  40. </ItemGroup>public void Initialize(GeneratorInitializationContext context)<ItemGroup>
  41.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  42.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  43. </ItemGroup>{<ItemGroup>
  44.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  45.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  46. </ItemGroup>    // No initialization required for this one<ItemGroup>
  47.     <PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.8.0" PrivateAssets="all" />
  48.     <PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4" PrivateAssets="all" />
  49. </ItemGroup>}    }}
复制代码
 
现在,我们有一个正常运行的生成器,但需要将其引用到控制台应用程序。 
新引用不是传统的项目引用,必须手动编辑以包含 OutputItemType 和 ReferenceOutputAssembly 属性。
  1. <ItemGroup>
  2.     <ProjectReference Include="..\SourceGeneratorInCSharp\SourceGeneratorInCSharp.csproj"
  3.                         OutputItemType="Analyzer"
  4.                         ReferenceOutputAssembly="false" />
  5. </ItemGroup>
复制代码
 
现在,运行控制台应用程序时,应会看到生成的代码运行并打印到屏幕。
控制台应用程序本身不实现 HelloFrom 方法,而是在编译过程中从源生成器项目生成的源。

 
如果使用的是 Visual Studio,则可以看到源生成的文件。
在“解决方案资源管理器”窗口中,展开“依赖项”>“分析器”>“SourceGenerator”>“SourceGenerator.HelloSourceGenerator”,然后双击“Program.g.cs”文件即可看到生成的内容。

 
https://learn.microsoft.com/zh-cn/dotnet/csharp/roslyn-sdk/source-generators-overview
 

来源:https://www.cnblogs.com/chenyishi/p/18073694
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具