- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试使用面向方面的编程来执行一个简单的斐波那契函数并跟踪对任何除了 Java 中的方法外,还显示嵌套他们的水平。
Java 代码:
package tracing;
public class Test {
static int fib(int n) {
if (n<=1)
return n;
else
return fib(n-1) + fib(n-2);
}
static void report(int n,int r) {
System.out.println("fib("+n+")="+r);
}
public static void main(String[] a) {
report(4,fib(4));
}
}
AspectJ 代码:
package tracing;
public aspect Trace {
String prefix = "";
Object around(): call(* *(..)) && !within(java..*) && !within(FigureEditor..*) {
System.out.println(prefix+thisJoinPoint.getSignature());
prefix = ">" + prefix;
Object result = proceed();
prefix = prefix.substring(1);
return result;
}
}
注意:&& !within(FigureEditor..*)
只是为了避免不同包的类中的函数。
控制台输出 - 错误:
Exception in thread "main" java.lang.StackOverflowError at
org.aspectj.runtime.internal.AroundClosure.<init>(AroundClosure.java:34)
at tracing.Trace$AjcClosure1.<init>(Trace.aj:1) at
tracing.Trace.ajc$around$tracing_Trace$1$ef88057b(Trace.aj:7) at
tracing.Trace.ajc$around$tracing_Trace$1$ef88057b(Trace.aj:7) at
tracing.Trace.ajc$around$tracing_Trace$1$ef88057b(Trace.aj:7) at
tracing.Trace.ajc$around$tracing_Trace$1$ef88057b(Trace.aj:7) at
tracing.Trace.ajc$around$tracing_Trace$1$ef88057b(Trace.aj:7) at
tracing.Trace.ajc$around$tracing_Trace$1$ef88057b(Trace.aj:7)
更新:我想要的输出类似于以下内容:
void Test.main(String [])
>void Test.report(int, int)<br>
>>int Test.fib(int)<br>
>>>int Test.fib(int)<br>
>>>>int Test.fib(int)<br>
>>>>>int Test.fib(int)<br>
>>>>>int Test.fib(int)<br>
>>>>int Test.fib(int)<br>
>>>int Test.fib(int)<br>
>>>>int Test.fib(int)<br>
>>>>int Test.fib(int)<br>
>> void Test.write(String) fib(4)=3
最佳答案
我不确定您是否正在使用 Eclipse 或其他一些 IDE 来开发您的 AspectJ 代码,但如果您这样做,您可能会注意到 IDE 在您的 的方法体中显示了 AspectJ 标记around()
建议本身,这意味着它们本身(您的 around()
建议中的方法调用)由您的方面编织而成,如下图所示:
这意味着,为了避免在您的 around()
通知中发生无限递归,在这种情况下,通知会从内部重复调用,您需要从中排除通知的控制流得到建议。您可以通过包含以下切入点表达式轻松地排除任何建议的控制流中的所有代码:!cflow(adviceexecution())
因此您的方面代码将如下所示(稍微重新格式化):
public aspect Trace {
String prefix = "";
Object around(): call(* *(..))
&& !within(java..*)
&& !within(FigureEditor..*)
&& !cflow(adviceexecution()) {
System.out.println(prefix+thisJoinPoint.getSignature());
prefix = ">" + prefix;
Object result = proceed();
prefix = prefix.substring(1);
return result;
}
}
来自AspectJ documentation on pointcut expressions :
cflow(Pointcut)
: Picks out each join point in the control flow of any join point P picked out by Pointcut, including P itself.
adviceexecution()
: Picks out all advice execution join points.
编辑:添加方面代码以反射(reflect)更新后的问题
您对原始问题的更新阐明了您的意图。我已经更新了方面代码以反射(reflect)这些变化。结果方面将如下所示:
public aspect Trace {
String prefix = "";
Object around(): call(* *(..))
&& !within(java..*)
&& !if(thisJoinPoint.getTarget()!=null && thisJoinPoint.getTarget().getClass().getName().startsWith("java"))
&& !within(FigureEditor..*)
&& !within(Trace) {
System.out.println(prefix+thisJoinPoint.getSignature());
prefix = ">" + prefix;
Object result = proceed();
prefix = prefix.substring(1);
return result;
}
}
我将所有建议执行代码的控制流排除替换为仅排除 Trace
方面,并包含一个运行时测试以排除对 java< 中代码的所有调用
包。
关于java - AOP - 错误 : java. lang.StackOverflowError 在 org.aspectj.runtime.internal.AroundClosure,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35486395/
我的问题与 Spring 的 AspectJ 模式有关,特别是如何启用它: 交易管理 缓存 1)我注意到,为了启用 AspectJ 模式进行事务管理,我只需执行以下操作: @Configuration
当我尝试使用 Java 17 运行 AspectJ 检测时,我总是会遇到如下错误: java.lang.reflect.InvocationTargetException at jav
我有一个应该记录的跟踪方面: 进入 退出(返回类型为 void) 返回[返回对象] 抛出[异常消息] 我对第二个有问题。我如何在不重复记录所有退出的情况下为这种情况创建建议,就像现在我有一个 @Aft
我已经使用 @Aspect 注释声明了我的切面,但建议似乎没有得到应用。该方面适用于我拥有的其他一些项目,主要区别似乎是其他项目完全使用注释连接,并且这个特定项目是 xml 连接的。唯一连接注释的 b
我正在尝试使用加载时编织将 perf4j 绑定(bind)到程序中,但它似乎在我的类路径中找不到 aop.xml。要么是这样,要么它没有编织这个方面,因为它没有找到它。我已启用 aop.xml 的详细
我是 spring 框架的新手,正在尝试一些示例来理解 AOP,这是我到目前为止所做的,但它不起作用。 问题是我一添加 对于 spring.xml,我的构建失败说无法创建具有空指针异常的 bean。但
我尝试使用 AspectJ 围绕 Kotlin 函数编织方面,但没有成功。也许我只是配置不正确,或者 AspectJ 不支持这个。 有谁知道这是否可以使用例如 maven 和 Eclipse(或 In
我正在使用 Eclipse 4 和 AspectJ 的最新版本进行开发。我正在用修改后的库(二进制编织)替换 Java 6 库。问题是当前正在编织的代码是 Java 7 代码,而我需要它是 Java
我正在将我的项目从 java 7 迁移到 java 8,我遇到的问题与使用 aspectj-maven-plugin 的 aspectj 编织有关。 我可以根据 Haus documentation
嘿,我想将 AOP 添加到我的 Web 项目中。我下载了 eclipse 3.4.1 的 ajdt2.0.1。但是当我将此项目转换为 AspectJ 项目时,出现了很多不应该发生的编译错误。比如“XX
我最近在我的 Windows 7 机器上从 eclipse Juno 升级到 Luna,我的 aspectj 编译出现问题。我收到此错误: [ERROR] Failed to execute goal
我想创建一个注释,它使用环绕方面来使用该注释清理参数。 例如,一个方法可能如下所示: public void setName(@Scrubbed String name) { ... } 也许 pub
我无法理解aspectJ的编译时和加载时编织,也无法弄清楚使用什么(以及如何使用ajc)来编译和构建我的项目。 这是我的项目结构:- TestProject:一个java服务库。这正被其他一些人使
我想拦截给定类的任何子类的非注释方法的执行。 例如,假设我有类 Base: public class Base { public void baseMethod() { //shouldn't
我正在尝试使用 AspectJ 和运行时编织。我创建了一个方面 @Aspect(value = "TraceAspect") public class TraceAspect { @Arou
我只是像下面描述的那样实现了AspectJ:https://stackoverflow.com/a/10998044/2182503 此解决方案工作正常,直到我注意到@Autowired中的@Init
我正在使用 AspectJ 来建议所有具有所选类参数的公共(public)方法。我尝试了以下方法: pointcut permissionCheckMethods(Session sess) :
我正在尝试创建一个 AspectJ Aspect 来拦截具有通用接口(interface)的返回方法。 这是我的 AspectJ @AspectJ public class MyAspect {
使用 Aspect annotation 创建方面时如所述 here ,是否可以将此注释与包含状态的类一起使用(例如,一旦命中切入点,成员变量就会发生变化)?或者换句话说:方面类是单例吗?注释的源代码
当我尝试使用 Roo 创建的 JPA 对象时出现此错误。 Entity manager has not been injected (is the Spring Aspects JAR configu
我是一名优秀的程序员,十分优秀!