使用 Aspect annotation 创建方面时如所述 here ,是否可以将此注释与包含状态的类一起使用(例如,一旦命中切入点,成员变量就会发生变化)?或者换句话说:方面类是单例吗?注释的源代码提出了类似的建议,但我在文档中找不到更多详细信息。
一个例子:这是一个计算方法被调用频率的方面。方面将调用次数存储在 map 中并打印出 map :
@Aspect
public class CountAspect {
private Map<String, Integer> counter = new HashMap<>();
@Before("execution(* *.*(..))")
public void countMethodCalls(JoinPoint jp) {
String methodName = jp.getSignature().getName();
counter.merge(methodName, 1, Integer::sum);
System.out.println(counter);
}
}
在测试这方面时,一切都按预期工作。调用方法时
a
, b
和 c
几次,输出是{main=1}
{a=1, main=1}
{a=1, b=1, main=1}
{a=1, b=1, c=1, main=1}
{a=2, b=1, c=1, main=1}
{a=3, b=1, c=1, main=1}
这个结果可靠吗?或者我的示例是否仅有效,因为我这里有一个相当简单的示例?
在更复杂的场景中,
CountAspect
的第二个实例是否会发生?将由 AspectJ 运行时创建,因此我会丢失在 counter
中收集的数据。 map 到那时?如果重要的话,我正在使用加载时编织。
最佳答案
AspectJ 知道多个实例化模型,其中之一(也是默认的)是单例模型。您可以在 AspectJ manual 中找到更多信息.在那里您将看到以下安装模型:
perthis
:每个 一个方面实例来电每个匹配切入点的实例 pertarget
:每个 一个方面实例被叫方每个匹配切入点的实例 percflow
:每输入一个方面实例 控制流 percflowbelow
:每输入一个方面实例 截获的控制流下方 (被拦截方法调用的一切)还描述了另一种实例化模型 here :
pertypewithin
:每个 一个方面实例类型(类)由切入点 所以是的,示例代码中显示的方面是单例,您可以依赖它。无论您是使用编译时编织、二进制编织还是加载时编织都没有关系,除非在应用服务器等容器中您有意通过类加载器隔离等方式创建不同的编织器实例。
关于java - AspectJ 方面是单例吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59523385/