Groovy简介——编译时方法注入

2024-04-02 05:58

本文主要是介绍Groovy简介——编译时方法注入,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一般只用于编写一些插件或者模板方法的时候使用,编译后的代码还是会合成为静态代码,会比运行时的处理方式效率更高。需要根据AST来添加对应的语法树。Groovy编译器允许我们进入其变异截断,一窥其所处理的AST(Abstract Syntax Tree 抽象语法树)。

 

AST:

这就是生成的语法树:

 

Groovy支持开发者在任何阶段介入:初始化、解析、转换、语义分析、规范化、指令选择、class生成、输出和结束

AST在语义分析阶段之后生成,如果想使用信息更多的AST,可以再之后的阶段介入。

 

CodeCheckpackage CodeAnalysisimport org.codehaus.groovy.ast.ASTNodeimport org.codehaus.groovy.ast.ClassNodeimport org.codehaus.groovy.ast.ConstructorNodeimport org.codehaus.groovy.ast.FieldNodeimport org.codehaus.groovy.ast.GroovyClassVisitorimport org.codehaus.groovy.ast.MethodNodeimport org.codehaus.groovy.ast.PropertyNodeimport org.codehaus.groovy.control.CompilePhaseimport org.codehaus.groovy.control.SourceUnitimport org.codehaus.groovy.syntax.SyntaxExceptionimport org.codehaus.groovy.transform.ASTTransformationimport org.codehaus.groovy.transform.GroovyASTTransformation@GroovyASTTransformation(phase = CompilePhase.SEMANTIC_ANALYSIS)class CodeCheck implements ASTTransformation {@Overridevoid visit(ASTNode[] astNodes, SourceUnit sourceUnit) {sourceUnit.ast.classes.each { classNode ->classNode.visitContents(new OurClassVisitor(sourceUnit))}}}class OurClassVisitor implements GroovyClassVisitor{SourceUnit sourceUnitOurClassVisitor(theSourceUnit){sourceUnit = theSourceUnit}@Overridevoid visitClass(ClassNode classNode) {}@Overridevoid visitConstructor(ConstructorNode constructorNode) {}private void reportError(message, lineNumber, columnNumber){sourceUnit.addError(new SyntaxException(message, lineNumber, columnNumber))}@Overridevoid visitMethod(MethodNode methodNode) {if (methodNode.name.size() == 1){reportError "Make method name descriptive, avoid single letter names",methodNode.lineNumber, methodNode.columnNumber}methodNode.parameters.each {parameter ->if (parameter.name.size() == 1){reportError "Single letter parameters are morally wrong!",parameter.lineNumber, parameter.columnNumber}}}@Overridevoid visitField(FieldNode fieldNode) {}@Overridevoid visitProperty(PropertyNode propertyNode) {}}

 

OurClassVisitor

 

package CodeAnalysisimport org.codehaus.groovy.ast.ASTNodeimport org.codehaus.groovy.ast.ClassNodeimport org.codehaus.groovy.ast.ConstructorNodeimport org.codehaus.groovy.ast.FieldNodeimport org.codehaus.groovy.ast.GroovyClassVisitorimport org.codehaus.groovy.ast.MethodNodeimport org.codehaus.groovy.ast.PropertyNodeimport org.codehaus.groovy.control.CompilePhaseimport org.codehaus.groovy.control.SourceUnitimport org.codehaus.groovy.syntax.SyntaxExceptionimport org.codehaus.groovy.transform.ASTTransformationimport org.codehaus.groovy.transform.GroovyASTTransformation@GroovyASTTransformation(phase = CompilePhase.SEMANTIC_ANALYSIS)class CodeCheck implements ASTTransformation {@Overridevoid visit(ASTNode[] astNodes, SourceUnit sourceUnit) {sourceUnit.ast.classes.each { classNode ->classNode.visitContents(new OurClassVisitor(sourceUnit))}}}class OurClassVisitor implements GroovyClassVisitor{SourceUnit sourceUnitOurClassVisitor(theSourceUnit){sourceUnit = theSourceUnit}@Overridevoid visitClass(ClassNode classNode) {}@Overridevoid visitConstructor(ConstructorNode constructorNode) {}private void reportError(message, lineNumber, columnNumber){sourceUnit.addError(new SyntaxException(message, lineNumber, columnNumber))}@Overridevoid visitMethod(MethodNode methodNode) {if (methodNode.name.size() == 1){reportError "Make method name descriptive, avoid single letter names",methodNode.lineNumber, methodNode.columnNumber}methodNode.parameters.each {parameter ->if (parameter.name.size() == 1){reportError "Single letter parameters are morally wrong!",parameter.lineNumber, parameter.columnNumber}}}@Overridevoid visitField(FieldNode fieldNode) {}@Overridevoid visitProperty(PropertyNode propertyNode) {}}

配置文件

 

manifest/META-INF/services/org.codehaus.groovy.transform.ASTTransformation

groovyc -classpath checkcode.jar smelly.groovy    

 

groovyc -d classes CodeAnalysis/CodeCheck.groovy  

jar -cf checkcode.jar -C classes CodeAnalysis -C manifest .

groovyc -classpath checkcode.jar smelly.groovy    

 

 

新建InjectAudit.groovy


 

package AST.InterceptingCalls.com.aglledeveloperimport org.codehaus.groovy.ast.ASTNodeimport org.codehaus.groovy.ast.expr.ArgumentListExpressionimport org.codehaus.groovy.ast.expr.MethodCallExpressionimport org.codehaus.groovy.ast.expr.VariableExpressionimport org.codehaus.groovy.ast.stmt.ExpressionStatementimport org.codehaus.groovy.control.CompilePhaseimport org.codehaus.groovy.control.SourceUnitimport org.codehaus.groovy.transform.ASTTransformationimport org.codehaus.groovy.transform.GroovyASTTransformation@GroovyASTTransformation(phase = CompilePhase.SEMANTIC_ANALYSIS)class InjectAudit implements ASTTransformation {@Overridevoid visit(ASTNode[] astNodes, SourceUnit sourceUnit) {def checkingAccountClassNode =astNodes[0].classes.find {it.name == 'AST.InterceptingCalls.CheckingAccount'}injectAuditMethod(checkingAccountClassNode)}static void injectAuditMethod(checkingAccountClassNode){def nonAuditMethods =checkingAccountClassNode?.methods.findAll{ it.name != 'audit'}nonAuditMethods?.each { injectMethodWithAudit(it) }}static void injectMethodWithAudit(methodNode){def callToAudit = new ExpressionStatement(new MethodCallExpression(new VariableExpression('this'),'audit',new ArgumentListExpression(methodNode.parameters)))methodNode.code.statements.add(0, callToAudit)}}

 

groovyc -d classes AST/InterceptingCalls/com/aglledeveloper/InjectAudit.groovy    

jar -cf injectAudit.jar -C classes AST -C manifest .

生成对应的jar包

 

UsingCheckingAccount.groovypackage AST.InterceptingCallsclass CheckingAccount {def audit(amount) {if (amount > 10000){print "auditing ..."}}def deposit(amount){println "depositing ${amount} ..."}def withdraw(amount){println "withdrawing ${amount} ..."}}def account = new CheckingAccount()account.deposit(1000)account.deposit(12000)account.withdraw(11000)groovy src/groovy/AST/InterceptingCalls/UsingCheckingAccount.groovydepositing 1000 ...depositing 12000 ...withdrawing 11000 ...depositing 1000 ...auditing ...depositing 12000 ...auditing ...withdrawing 11000 ...

 

执行:

groovyc -d classes src/groovy/AST/InterceptingCalls/com/aglledeveloper/InjectAudit.groovy

jar -cf injectAudit.jar -C classes AST -C manifest .    

groovy -classpath injectAudit.jar src/groovy/AST/InterceptingCalls/UsingCheckingAccount.groovy

 

 

结果:

depositing 1000 ...

auditing ...depositing 12000 ...

auditing ...withdrawing 11000 ...

 

*audit方法被应用在最前方,当对于10000的时候会自动审计

 

 

这篇关于Groovy简介——编译时方法注入的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/869159

相关文章

Python常用命令提示符使用方法详解

《Python常用命令提示符使用方法详解》在学习python的过程中,我们需要用到命令提示符(CMD)进行环境的配置,:本文主要介绍Python常用命令提示符使用方法的相关资料,文中通过代码介绍的... 目录一、python环境基础命令【Windows】1、检查Python是否安装2、 查看Python的安

Maven 配置中的 <mirror>绕过 HTTP 阻断机制的方法

《Maven配置中的<mirror>绕过HTTP阻断机制的方法》:本文主要介绍Maven配置中的<mirror>绕过HTTP阻断机制的方法,本文给大家分享问题原因及解决方案,感兴趣的朋友一... 目录一、问题场景:升级 Maven 后构建失败二、解决方案:通过 <mirror> 配置覆盖默认行为1. 配置示

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

Mybatis Plus Join使用方法示例详解

《MybatisPlusJoin使用方法示例详解》:本文主要介绍MybatisPlusJoin使用方法示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,... 目录1、pom文件2、yaml配置文件3、分页插件4、示例代码:5、测试代码6、和PageHelper结合6

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

SpringBoot读取ZooKeeper(ZK)属性的方法实现

《SpringBoot读取ZooKeeper(ZK)属性的方法实现》本文主要介绍了SpringBoot读取ZooKeeper(ZK)属性的方法实现,强调使用@ConfigurationProperti... 目录1. 在配置文件中定义 ZK 属性application.propertiesapplicati

MyBatis设计SQL返回布尔值(Boolean)的常见方法

《MyBatis设计SQL返回布尔值(Boolean)的常见方法》这篇文章主要为大家详细介绍了MyBatis设计SQL返回布尔值(Boolean)的几种常见方法,文中的示例代码讲解详细,感兴趣的小伙伴... 目录方案一:使用COUNT查询存在性(推荐)方案二:条件表达式直接返回布尔方案三:存在性检查(EXI