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

相关文章

SQL注入漏洞扫描之sqlmap详解

《SQL注入漏洞扫描之sqlmap详解》SQLMap是一款自动执行SQL注入的审计工具,支持多种SQL注入技术,包括布尔型盲注、时间型盲注、报错型注入、联合查询注入和堆叠查询注入... 目录what支持类型how---less-1为例1.检测网站是否存在sql注入漏洞的注入点2.列举可用数据库3.列举数据库

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

Git中恢复已删除分支的几种方法

《Git中恢复已删除分支的几种方法》:本文主要介绍在Git中恢复已删除分支的几种方法,包括查找提交记录、恢复分支、推送恢复的分支等步骤,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录1. 恢复本地删除的分支场景方法2. 恢复远程删除的分支场景方法3. 恢复未推送的本地删除分支场景方法4. 恢复

Python将大量遥感数据的值缩放指定倍数的方法(推荐)

《Python将大量遥感数据的值缩放指定倍数的方法(推荐)》本文介绍基于Python中的gdal模块,批量读取大量多波段遥感影像文件,分别对各波段数据加以数值处理,并将所得处理后数据保存为新的遥感影像... 本文介绍基于python中的gdal模块,批量读取大量多波段遥感影像文件,分别对各波段数据加以数值处

Window Server2016加入AD域的方法步骤

《WindowServer2016加入AD域的方法步骤》:本文主要介绍WindowServer2016加入AD域的方法步骤,包括配置DNS、检测ping通、更改计算机域、输入账号密码、重启服务... 目录一、 准备条件二、配置ServerB加入ServerA的AD域(test.ly)三、查看加入AD域后的变

Window Server2016 AD域的创建的方法步骤

《WindowServer2016AD域的创建的方法步骤》本文主要介绍了WindowServer2016AD域的创建的方法步骤,文中通过图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、准备条件二、在ServerA服务器中常见AD域管理器:三、创建AD域,域地址为“test.ly”

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服

Golang的CSP模型简介(最新推荐)

《Golang的CSP模型简介(最新推荐)》Golang采用了CSP(CommunicatingSequentialProcesses,通信顺序进程)并发模型,通过goroutine和channe... 目录前言一、介绍1. 什么是 CSP 模型2. Goroutine3. Channel4. Channe

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

Python中使用defaultdict和Counter的方法

《Python中使用defaultdict和Counter的方法》本文深入探讨了Python中的两个强大工具——defaultdict和Counter,并详细介绍了它们的工作原理、应用场景以及在实际编... 目录引言defaultdict的深入应用什么是defaultdictdefaultdict的工作原理