.NET Core技术研究-通过Roslyn代码分析技术规范提升代码质量

随着团队越来越多,越来越大,需求更迭越来越快,每天提交的代码变更由原先的2位数,暴涨到3位数,每天几百次代码Check In,补丁提交,大量的代码审查消耗了大量的资源投入。

如何确保提交代码的质量和提测产品的质量,这两个是非常大的挑战。

工欲善其事,必先利其器。在上述需求背景下,今年我们准备用工具和技术,全面把控并提升代码质量和产品提测质量。即:

1. 代码质量提升:通过自定义代码扫描规则,将有问题的代码、不符合编码规则的代码扫描出来,禁止签入

2. 产品提测质量:通过单元测试覆盖率和执行通过率,严控产品提交质量,覆盖率和通过率达不到标准,无法提交测试。

准备用2篇文章,和大家分享我们是如何提升代码质量和产品提测质量的。今天分享第一篇:通过Roslyn代码分析全面提升代码质量。

一、什么是Roslyn

  Roslyn 是微软开源的 .NET 编译平台(.NET Compiler Platform)。  编译平台支持 C# 和 Visual Basic 代码编译,并提供丰富的代码分析 API。

  利用Roslyn可以生成代码分析器和代码修补程序,从而发现和更正编码错误。 

  分析器不仅理解代码的语法和结构,还能检测应更正的做法。代码修补程序建议一处或多处修复,以修复分析器发现的编码错误。

  我们写下面一堆代码,Roslyn编译器会有如下提示: 

  

 通过编写分析器和代码修补程序,主要服务以下场景:  

  • 强制执行团队编码标准(Local)

  • 提供库包方面的指导约束(Nuget)

  • 提供代码分析器相关的VSIX扩展插件(Visual Studio Marketplace)

 Roslyn是如何做到代码分析的呢?这背后依赖于一套强大的语法分析和API:

 

  上图中:Language Service:语言层面的服务,可以简单理解为我们在VS中编码时,可以实现的语法高亮、查找所有引用、重命名、转到定义、格式化、抽取方法等操作

  Compiler API:编译器API,这里提供了Syntax Tree API代码语法树API,Symbol API代码符号API

  Binding and Flow Anllysis APIs绑定和流分析API(https://joshvarty.com/2015/02/05/learn-roslyn-now-part-8-data-flow-analysis/),

  Emit API编译反射发出API(https://joshvarty.com/2016/01/16/learn-roslyn-now-part-16-the-emit-api/)

  这里我们详细看一下语法树、符号、语义模型、工作区:

  1. 语法树是一种由编译器 API 公开的基础数据结构。 这些树表示源代码的词法和语法结构。其包含:  

  • 语法节点:是语法树的一个主要元素。这些节点表示声明、语句、子句和表达式等语法构造。

  • 语法标记:表示代码的最小语法片段。语法标记包含关键字、标识符、文本和标点。

  • 琐碎内容:对正常理解代码基本上没有意义的源文本部分,例如空格、注释和预处理器指令。

  • 范围:每个节点、标记或琐碎内容在源文本内的位置和包含的字符数。

  • 种类:标识节点、标记或琐碎内容所表示的确切语法元素。

  • 错误:表示源文本中包含的语法错误。

     看一张语法树的图:

  

  2. 符号:符号表示源代码声明的不同元素,或作为元数据从程序集中导出。每个命名空间、类型、方法、属性、字段、事件、参数或局部变量都由符号表示。

  3. 语义模型:语义模型表示单个源文件的所有语义信息。可使用语义模型查找到以下内容:   

  • 在源中特定位置引用的符号。

  • 任何表达式的结果类型。

  • 所有诊断(错误和警告)。

  • 变量流入和流出源区域的方式。

  • 更多推理问题的答案。

  4. 工作区:工作区是对整个解决方案执行代码分析和重构的起点。相关的API可以实现:

     将解决方案中项目的全部相关信息组织为单个对象模型,可让用户直接访问编译器层对象模型(如源文本、语法树、语义模型和编译),而无需分析文件、配置选项,或管理项目内依赖项。

   

  了解了Roslyn的大致情况之后,我们开始基于Roslyn做一些“不符合编程规范要求(团队自定义的)”的代码分析。

二、基于Roslyn进行代码分析

  接下来讲通过Show case的方法,通过实际的场景和大家分享。在我们编写实际的代码分析器之前,我们先把开发环境准备好  :

    使用VS2017创建一个Analyzer with Code Fix工程

    因为我本机的VS2019找了好久没找到对应的工程,这个章节,使用VS2017吧

    创建完成会有两个工程:

    

    其中,TeldCodeAnalyzer.Vsix工程,主要用以生成VSIX扩展文件

   TeldCodeAnalyzer工程,主要用于编写代码分析器。

    工程转换好之后,我们开始编码吧。

 1. catch 吞掉异常场景

  问题:catch吞掉异常后,线上很难排查问题,同时确定哪块代码有问题

  示例代码:

1

2

3

4

5

6

7

8

9

try

{

     var logService = HSFService.Proxy<ILogService>();

     logService.SendMsg(new SysActionLog());

}

catch (Exception ex)

{

                 

}

  需求:当开发人员在catch吞掉异常时,给与编程提示:异常吞掉时必须上报监控或者日志

  明确了上述需要,我们开始编写Roslyn代码分析器。ExceptionCatchWithMonitorAnalyzer

  

  我们详细解读一下:

  ① ExceptionCatchWithMonitorAnalyzer必须继承抽象类DiagnosticAnalyzer

  ② 重写方法SupportedDiagnostics,注册代码扫描规则:DiagnosticDescriptor    

1

2

3

4

internal static DiagnosticDescriptor Rule = new DiagnosticDescriptor(DiagnosticId, Title, MessageFormat, Category,

            DiagnosticSeverity.Warning, isEnabledByDefault: true, description: Description);

 

public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics => ImmutableArray.Create(Rule);

 ③ 重写方法Initialize,注册Microsoft.CodeAnalysis.SyntaxNode完成Catch语句的语义分析后的事件Action

1

2

3

4

5

public override void Initialize(AnalysisContext context)

{           context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.Analyze | GeneratedCodeAnalysisFlags.None);

            context.EnableConcurrentExecution();

            context.RegisterSyntaxNodeAction(AnalyzeDeclaration, SyntaxKind.CatchClause);

}

 ④ 实现语法分析AnalyzeDeclaration,检查对catch语句中代码实现   

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

private void AnalyzeDeclaration(SyntaxNodeAnalysisContext context)

{

            var catchClause = (CatchClauseSyntax)context.Node;

            var block = catchClause.Block;

            foreach (var statement in block.Statements)

            {

                if (statement is ThrowStatementSyntax)

                {

                    return;

                }

            }

 

 

            if (Common.IsReallyContains(block, "MonitorClient") == false)

            {

                context.ReportDiagnostic(Diagnostic.Create(Rule, block.GetLocation()));

            }

}

  代码实现后的效果(直接调试VSIX工程即可)

  

代码编译后也有对应Warnning提示

 2. 在For循环中进行服务调用

  问题:for循环中调用RPC服务,每次访问都会发起一次RPC请求,如果循环次数太多,性能很差,建议使用批量处理的RPC方法

  示例代码:


foreach (var item in items)

{

      var logService = HSFService.Proxy<ILogService>();

      logService.SendMsg(new SysActionLog());

}  

  需求:当开发人员在For循环中调用HSF服务时,给与编程提示:不建议在循环中调用HSF服务, 建议调用批量处理方法.

  明确了上述需要,我们开始编写Roslyn代码分析器。HSFForLoopAnalyzer  


[DiagnosticAnalyzer(LanguageNames.CSharp)]

public sealed class HSFForLoopAnalyzer : DiagnosticAnalyzer

{

    public const string DiagnosticId = "TA001";

    internal const string Title = "增加循环中HSF服务调用检查";

    public const string MessageFormat = "不建议在循环中调用HSF服务, 建议调用批量处理方法.";

    internal const string Category = "CodeSmell";

 

    internal static DiagnosticDescriptor Rule = new DiagnosticDescriptor(DiagnosticId, Title, MessageFormat, Category,

        DiagnosticSeverity.Warning, isEnabledByDefault: true);

 

    public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics => ImmutableArray.Create(Rule);

 

    public override void Initialize(AnalysisContext context)

    {

        context.RegisterSyntaxNodeAction(AnalyzeMethodForLoop, SyntaxKind.InvocationExpression);

    }

 

    private static void AnalyzeMethodForLoop(SyntaxNodeAnalysisContext context)

    {

        var expression = (InvocationExpressionSyntax)context.Node;

        string exressionText = expression.ToString();

        if (Common.IsReallyContains(expression, "HSFService.Proxy<"))

        {

            var loop = expression.Ancestors().FirstOrDefault(p => p is ForStatementSyntax || p is ForEachStatementSyntax || p is DoStatementSyntax || p is WhileStatementSyntax);

            if (loop != null)

            {

                var diagnostic = Diagnostic.Create(Rule, expression.GetLocation());

                context.ReportDiagnostic(diagnostic);

                return;

            }

 

            if (Common.IsReallyContains(expression, ">.") == false)

            {

                var syntax = expression.Ancestors().FirstOrDefault(p => p is LocalDeclarationStatementSyntax);

                if (syntax != null)

                {

                    var declaration = (LocalDeclarationStatementSyntax)syntax;

                    var variable = declaration.Declaration.Variables.SingleOrDefault();

                   

 

                    var method = declaration.Ancestors().First(p => p is MethodDeclarationSyntax);

                    var expresses = method.DescendantNodes().Where(p => p is InvocationExpressionSyntax);

                    foreach (var express in expresses)

                    {

                        loop = express.Ancestors().FirstOrDefault(p => p is ForStatementSyntax || p is ForEachStatementSyntax || p is DoStatementSyntax || p is WhileStatementSyntax);

                        if (loop != null)

                        {

                            var diagnostic = Diagnostic.Create(Rule, expression.GetLocation());

                            context.ReportDiagnostic(diagnostic);

                            return;

                        }

                    }

                }

            }

        }

    }

}

  基本的实现方式,和上一个差不多,唯一不同的逻辑是在实际的代码分析过程中,AnalyzeMethodForLoop。大家可以根据自己的需要写一下。

       实际的效果:

       

       还有几个代码检查场景,基本都是同样的实现思路,再次不一一罗列了。

       在这里还可以自动完成代理修补程序,这个地方我们还在研究中,可能每个业务代码的场景不同,很难给出一个通用的改进代码,所以这个地方等后续我们完成后,再和大家分享。

三、通过Roslyn实现静态代码扫描

  线上很多代码已经写完了,发布上线了,对已有的代码进行代码扫描也是非常重要的。因此,我们对catch吞掉异常的代码进行了一次集中扫描和改进。

  那么基于Roslyn如何实现静态代码扫描呢?主要的步骤有:

  ① 创建一个编译工作区MSBuildWorkspace.Create()

  ② 打开解决方案文件OpenSolutionAsync(slnPath);  

  ③ 遍历Project中的Document

  ④ 拿到代码语法树、找到Catch语句CatchClauseSyntax

  ⑤ 判断是否有throw语句,如果没有,收集数据进行通知改进

  看一下具体代码实现:

  先看一下Nuget引用:

  Microsoft.CodeAnalysis

  Microsoft.CodeAnalysis.Workspaces.MSBuild

  

  代码的具体实现:

      


public async Task<List<CodeCheckResult>> CheckSln(string slnPath)

       {

           var slnFile = new FileInfo(slnPath);

           var results = new List<CodeCheckResult>();         

           var solution = await MSBuildWorkspace.Create().OpenSolutionAsync(slnPath);           

 

           if (solution.Projects != null && solution.Projects.Count() > 0)

           {

               foreach (var project in solution.Projects.ToList())

               {

                   var documents = project.Documents.Where(x => x.Name.Contains(".cs"));

 

                   foreach (var document in documents)

                   {

                       var tree = await document.GetSyntaxTreeAsync();

                       var root = tree.GetCompilationUnitRoot();

                       if (root.Members == null || root.Members.Count == 0) continue;

                       //member

                       var firstmember = root.Members[0];

                       //命名空间Namespace

                       var namespaceDeclaration = (NamespaceDeclarationSyntax)firstmember;

 

                       foreach (var classDeclare in namespaceDeclaration.Members)

                       {

                           var programDeclaration = classDeclare as ClassDeclarationSyntax;

 

                           foreach (var method in programDeclaration.Members)

                           {

 

                               //方法 Method

                               var methodDeclaration = (MethodDeclarationSyntax)method;

 

                               var catchNode = methodDeclaration.DescendantNodes().FirstOrDefault(i => i is CatchClauseSyntax);

                               if (catchNode != null)

                               {

                                   var catchClause = catchNode as CatchClauseSyntax;

                                   if (catchClause != null || catchClause.Declaration != null)

                                   {

                                       if (catchClause.DescendantNodes().OfType<ThrowStatementSyntax>().Count() == 0)

                                       {

                                           results.Add(new CodeCheckResult()

                                           {

                                               Sln = slnFile.Name,

                                               ProjectName = project.Name,

                                               ClassName = programDeclaration.Identifier.Text,

                                               MethodName = methodDeclaration.Identifier.Text,

                                           });

                                       }

                                   }

                               }

                           }

                       }

                   }

               }

           }

 

           return results;

       }  

     以上是通过Roslyn代码分析全面提升代码质量的一些具体实践,分享给大家。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/310138.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python最大堆heapq_Python-堆的实现与heapq(最小堆库函数)

目录简介堆是一个二叉树&#xff0c;它的每个父节点的值都只会小于或大于所有孩子节点(的值)。它使用了数组来实现&#xff1a;从零开始计数&#xff0c;对于所有的 k &#xff0c;都有 heap[k] < heap[2*k1] 和 heap[k] < heap[2*k2]。 为了便于比较&#xff0c;不存在的…

深入浅出 ASP.NET Core 与 Docker 入门课程说明

点击蓝字“角落的白板报”关注我哟加个“星标★”&#xff0c;好文必达&#xff01;深入浅出 ASP.NET Core 与 Docker 入门课程说明《深入浅出 ASP.NET Core 与 Docker 》是一门新的课程&#xff0c;本课程所有的内容全部免费&#xff0c;以图文配合视频的形式呈现。课程完整视…

微软将在新西兰建设其第一个数据中心区域

昨天新西兰各IT群都被一条消息刷屏了&#xff1a;详情可见&#xff1a;https://news.microsoft.com/en-nz/2020/05/06/aotearoa-disclosure/NZ的第一个Azure region region 是云计算的一个术语&#xff0c;也就是各大云运营商机房部署的位置。目前微软、亚马逊、谷歌等比较大的…

使用 kind 快速搭建一个 Kubernetes 测试环境

使用 kind 快速搭建一个 Kubernetes 测试环境Introkind&#xff08;Kubernetes IN Docker&#xff09; 是一个基于 docker 构建 Kubernetes 集群的工具&#xff0c;非常适合用来在本地搭建基于 Kubernetes 的开发/测试环境。想写一篇 kind 的文章很久了&#xff0c;但是之前的 …

麻雀虽小,五脏俱全

入职三年&#xff0c; 除了参与公司核心产品研发外&#xff0c;另外负责了一个2C的小项目&#xff1a;调用API拿到解析结果 & 计费。❝项目最初是.NetCore 1.0-Previewsqlite部署在IIS上&#xff0c;闲来没事&#xff0c;这个项目已经被我完全重写&#xff0c;在此记录一些…

内存迟迟下不去,可能你就差一个GC.Collect

一&#xff1a;背景1. 讲故事我们有一家top级的淘品牌店铺&#xff0c;为了后续的加速计算&#xff0c;在程序启动的时候灌入她家的核心数据到内存中&#xff0c;灌入完成后内存高达100G&#xff0c;虽然云上的机器内存有256G&#xff0c;然被这么划掉一半看着还是有一点心疼的…

Mayor's posters POJ - 2528 (离散化+线段树)

题意&#xff1a; 在1~10000000这个区间中读取n个海报的区间信息&#xff0c;后面的海报会覆 盖前面的海报&#xff0c;问最后能看到几张海报.&#xff08;本题是一道bug题下面会提&#xff09; 题目&#xff1a; The citizens of Bytetown, AB, could not stand that the c…

[PAT乙级]1018 锤子剪刀布

大家应该都会玩“锤子剪刀布”的游戏&#xff1a;两人同时给出手势&#xff0c;胜负规则如图所示&#xff1a; 现给出两人的交锋记录&#xff0c;请统计双方的胜、平、负次数&#xff0c;并且给出双方分别出什么手势的胜算最大。 输入格式&#xff1a; 输入第 1 行给出正整数 …

java 开源控件_一些好用的开源控件

工作两年&#xff0c;一直都在做些编码方面的表面功夫&#xff0c;实现了很多很炫的功能&#xff0c;在此写下一些体验。有些比较小的dll文件我会发上来&#xff0c;如果是开源组织的代码我会把地址附上&#xff0c;毕竟人家是会更新的。大家还有什么好用的开源控件欢迎补充。一…

揭秘 .NET 5 和Java 互操作

早早的.NET团队就立下了.NET和Java互操作的flag&#xff0c;如果你去翻一翻dotnet/runtime库&#xff0c;丝毫看不出来仓库内在搞支持。xamarin/java.interop库一直有Mono和Java互操作的实现&#xff0c;那么100%的实现.NET和Java互操作就是它&#xff0c;这两篇文章就是和你一…

java中的asList_Java Arrays.AsList原理及用法实例

java.util.Arrays的asList方法可以方便的将数组转化为集合&#xff0c;我们平时开发在初始化ArrayList时使用的比较多&#xff0c;可以简化代码&#xff0c;但这个静态方法asList()有几个坑需要注意:一. 如果对集合使用增加或删除元素的操作将会报错如下代码&#xff1a;List l…

为什么顶尖高手,都是长期主义者?

点击蓝字关注&#xff0c;回复“职场进阶”获取职场进阶精品资料一份在当下这个浮躁的社会&#xff0c;很多人都认为“快”才是真理&#xff0c;“慢”就是原罪。大家都对一夜腾飞的故事津津乐道&#xff0c;却觉得坚守一生最终获得大成的故事不够性感。但事实是怎么样的呢&…

java中函数是什么_[一] java8 函数式编程入门 什么是函数式编程 函数接口概念 流和收集器基本概念...

本文是针对于java8引入函数式编程概念以及stream流相关的一些简单介绍什么是函数式编程?java程序员第一反应可能会理解成类的成员方法一类的东西此处并不是这个含义,更接近是数学上的函数看一下百度百科中关于函数的说明函数的定义&#xff1a;给定一个数集A&#xff0c;假设其…

分布式系统不得不说的CAP定理

21天学会C语言&#xff1f;3天学会弹钢琴&#xff1f;放弃一切错误方法&#xff0c;从今天开始“刻意练习”&#xff0c;因为这才是最强大的&#xff0c;也是唯一正确的学习方法。--《刻意练习》Anders Ericsson引言CAP问题已经成了计算机科学中一个研究领域&#xff0c;之前说…

[PAT乙级]1021 个位数统计

输入格式&#xff1a; 每个输入包含 1 个测试用例&#xff0c;即一个不超过 1000 位的正整数 N。 输出格式&#xff1a; 对 N 中每一种不同的个位数字&#xff0c;以 D:M 的格式在一行中输出该位数字 D 及其在 N 中出现的次数 M。要求按 D 的升序输出。 输入样例&#xff1a;…

Largest Rectangle in a Histogram (动态规划+奇思妙想单调栈)求最大矩状图面积

感觉动态规划都是玄妙的很&#xff0c;思维题吧&#xff08;单调栈思维&#xff09; 题解&#xff1a;让求最大矩形面积&#xff0c;宽为1&#xff0c;暴力超时 可以发现 当第i-1个比第i个高的时候 比第i-1个高的所有也一定比第i个高 于是可以用到动态规划的思想 令l…

ASP.NET Core分布式项目实战(详解oauth2授权码流程)--学习笔记

最近公司产品上线&#xff0c;通宵加班了一个月&#xff0c;一直没有更新&#xff0c;今天开始恢复&#xff0c;每日一更&#xff0c;冲冲冲任务13&#xff1a;详解oauth2授权码流程我们即将开发的产品有一个用户 API&#xff0c;一个项目服务 API&#xff0c;每个服务都需要认…

C++类模板中的static成员

从类模板实例化的每一个模板类有自己的类模板数据成员&#xff0c;该模板的所有对象共享一个static数据成员。 代码如下: #include <iostream> using namespace std;template<typename T> class Person { public:static int a;};template<typename T> int …

C#并发编程之初识并行编程

写在前面之前微信公众号里有一位叫sara的朋友建议我写一下Parallel的相关内容&#xff0c;因为手中商城的重构工作量较大&#xff0c;一时之间无法抽出时间。近日&#xff0c;这套系统已有阶段性成果&#xff0c;所以准备写一下Parallel的相关内容&#xff0c;正好也延续之前的…

java 下拉列表 枚举_「Java三分钟」精准而优雅——枚举类详解

关注我&#xff0c;每天三分钟&#xff0c;带你轻松掌握一个Java相关知识点。1.为什么要用枚举你在读一个老工程代码时&#xff0c;是否经常看见有几个类&#xff0c;里面放着成百上千的静态常量&#xff0c;场面相当恐怖&#xff0c;而且如果不加注释&#xff0c;很多你都不知…