Max*_*xym 55
cflow可以帮助您建议整个控制流程.我们试试一个例子,我有4个小班
public class A {
public static void methodA() {
B.methodB();
}
}
public class B {
public static void methodB() {
C.methodC();
int a = 1;
int b = 2;
System.out.println( a + b );
}
}
public class C {
public static void methodC() {
D.methodD();
}
}
public class D {
public static void methodD() {
}
}
Run Code Online (Sandbox Code Playgroud)
我的方面:
public aspect CFlow {
public pointcut flow() : cflow(call( * B.methodB() ) ) && !within(CFlow);
before() : flow() {
System.out.println( thisJoinPoint );
}
}
Run Code Online (Sandbox Code Playgroud)
和我的跑步者(只是为了看看会发生什么):
public class Test {
public static void main(String[] args) {
A.methodA();
}
}
Run Code Online (Sandbox Code Playgroud)
在我的切入点你可以看到cflow(call( * B.methodB() ) ),所以我想从B.methodB调用开始控制流程,当你运行Test类时,你会在控制台上看到:
call(void test.B.methodB())
staticinitialization(test.B.<clinit>)
execution(void test.B.methodB())
call(void test.C.methodC())
staticinitialization(test.C.<clinit>)
execution(void test.C.methodC())
call(void test.D.methodD())
staticinitialization(test.D.<clinit>)
execution(void test.D.methodD())
get(PrintStream java.lang.System.out)
call(void java.io.PrintStream.println(int))
3
Run Code Online (Sandbox Code Playgroud)
最后一个字符串不属于方面,它只是因为System.out.println里面methodB.所有打印的节目都可以控制流程 - 方法链和"事件"(执行,调用,初始化......).你看,我是从Test课堂开始的,虽然methodA我们对methodB控制流程很感兴趣但是他们不在'堆栈'中.
如果你想获得那个堆栈,但没有第一行(自己调用),你可以尝试定义
public pointcut flow() : cflowbelow(call( * B.methodB() ) ) && !within(CFlow);
Run Code Online (Sandbox Code Playgroud)
cflowbelow是另一个切入点,这意味着控制流不包括指定(在我们的情况下调用B.methodB).
小心添加!within(_aspect_)切入点,否则你什么都不会好StackOverflowError.这是因为cflow不能在编译时定义,并且在运行时方面也属于控制流(因此它会导致永久递归...)
好吧,把控制流看作类似于调用堆栈,然后你就会知道它的用法;)