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

相关文章

Linux换行符的使用方法详解

《Linux换行符的使用方法详解》本文介绍了Linux中常用的换行符LF及其在文件中的表示,展示了如何使用sed命令替换换行符,并列举了与换行符处理相关的Linux命令,通过代码讲解的非常详细,需要的... 目录简介检测文件中的换行符使用 cat -A 查看换行符使用 od -c 检查字符换行符格式转换将

Java编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

macOS无效Launchpad图标轻松删除的4 种实用方法

《macOS无效Launchpad图标轻松删除的4种实用方法》mac中不在appstore上下载的应用经常在删除后它的图标还残留在launchpad中,并且长按图标也不会出现删除符号,下面解决这个问... 在 MACOS 上,Launchpad(也就是「启动台」)是一个便捷的 App 启动工具。但有时候,应

SpringBoot日志配置SLF4J和Logback的方法实现

《SpringBoot日志配置SLF4J和Logback的方法实现》日志记录是不可或缺的一部分,本文主要介绍了SpringBoot日志配置SLF4J和Logback的方法实现,文中通过示例代码介绍的非... 目录一、前言二、案例一:初识日志三、案例二:使用Lombok输出日志四、案例三:配置Logback一

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

mysql出现ERROR 2003 (HY000): Can‘t connect to MySQL server on ‘localhost‘ (10061)的解决方法

《mysql出现ERROR2003(HY000):Can‘tconnecttoMySQLserveron‘localhost‘(10061)的解决方法》本文主要介绍了mysql出现... 目录前言:第一步:第二步:第三步:总结:前言:当你想通过命令窗口想打开mysql时候发现提http://www.cpp

Mysql删除几亿条数据表中的部分数据的方法实现

《Mysql删除几亿条数据表中的部分数据的方法实现》在MySQL中删除一个大表中的数据时,需要特别注意操作的性能和对系统的影响,本文主要介绍了Mysql删除几亿条数据表中的部分数据的方法实现,具有一定... 目录1、需求2、方案1. 使用 DELETE 语句分批删除2. 使用 INPLACE ALTER T