- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
如果 AspectJ 的工作方式与 EJB 拦截器的工作方式相同,这个问题就不会存在。
考虑 EJB 拦截器方式的基本场景:
@AroundInvoke
public Object log(final InvocationContext ctx) throws Exception {
// do stuff before
final Object result = ctx.proceed();
// do stuff after
return result;
}
现在我需要被拦截的 Method 对象。在这里我可以简单地这样做:
Method method = ctx.getMethod();
就是这样,之后我将检查拦截方法的注释。
现在我正在使用部署到 servlet 容器 (Tomcat) 的应用程序,因此没有 EJB。其中AOP是使用Spring + AspectJ实现的。
around 方法如下所示:
@Around("execution(* foo.bar.domain.integration.database.dao..*(..))")
public Object log(final ProceedingJoinPoint pjp) throws Throwable {
// do stuff before
final Object result = pjp.proceed();
// do stuff after
return result;
}
在这里我不能再这样做:
Method method = pjp.getMethod(); // no such method exists
相反,我被迫使用反射自己获取 Method 对象,如下所示:
final String methodName = pjp.getSignature().getName();
final Object[] arguments = pjp.getArgs();
final Class[] argumentTypes = getArgumentTypes(arguments);
final Method method = pjp.getTarget().getClass().getMethod(methodName, argumentTypes);
它一直有效,直到您想要掌握模板方法为止:
@Transactional
public <T extends Identifiable> T save(T t) {
if (null == t.getId()) {
create(t);
return t;
}
return update(t);
}
假设您按如下方式调用此方法:
Person person = new Person("Oleg");
personService.save(person);
您将收到:
Caused by: java.lang.NoSuchMethodException: foo.bar.domain.integration.database.dao.EntityService.save(foo.bar.domain.entity.Person)
抛出者:
pjp.getTarget().getClass().getMethod()
问题是泛型在运行时不存在,实际的方法签名是:
public Identifiable save(Identifiable t) {}
final Class[] argumentTypes 将包含一个元素,其类型将为 Person。所以这个调用:
pjp.getTarget().getClass().getMethod(methodName, argumentTypes);
查找方法:
save(Person p)
并且找不到它。
所以问题是:如何获取 java 的 template Method 对象的实例来表示已被拦截的确切方法?
我想其中一种方法是用蛮力的方式来做到这一点:获取参数父类(super class)/接口(interface),并尝试使用这些类型获取方法,直到不再得到 NoSuchMethodException,但这实在是丑陋。有没有一种正常的干净方法可以做到这一点?
最佳答案
好的,我的问题现在已经解决了。仔细一看后发现:methodSignature.getMethod()返回我注意到它返回的是接口(interface)而不是实现类,当然接口(interface)上没有注释。这与 EJB 拦截器不同,EJB 拦截器的 getMethod() 返回实现类方法。
所以最终的解决方案是这样的:
final String methodName = pjp.getSignature().getName();
final MethodSignature methodSignature = (MethodSignature)pjp.getSignature();
Method method = methodSignature.getMethod();
if (method.getDeclaringClass().isInterface()) {
method = pjp.getTarget().getClass().getDeclaredMethod(methodName, method.getParameterTypes());
}
如果您愿意,您也可以在需要时在这里处理接口(interface)注释。
还要注意这一点:method.getParameterTypes()如果没有这个,它仍然会抛出NoSuchMethodException,所以我们可以通过 ((MethodSignature)pjp.getSignature()).getMethod(); 获得正确的签名,这是一件好事。
希望不会再有什么意外,尽管我不喜欢在这里使用反射,但我只是更喜欢拥有实现类的方法实例,就像 EJB 的 InitationContext 中那样。
顺便说一句,Spring 的原生方法没有 AspectJ:
public Object invoke(final MethodInvocation invocation) throws Throwable {}
返回接口(interface)而不是实现类。为了长知识,也检查了一下。
最诚挚的问候和感谢您的帮助,我真的很感激。奥列格
关于java - 从 org.aspectj.lang.ProceedingJoinPoint 获取模板/通用 java.lang.reflect.Method 对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5212088/
在 Tomcat 6/Ubuntu 12.04 上启动 Grails 2.1.0 应用程序时出现以下错误。 Error 500 - Internal Server Error. groovy.lang
在运行 Storm 拓扑时,我收到此错误。拓扑完美运行 5 分钟,没有任何错误,然后失败。我正在使用 Config.TOPOLOGY_TICK_TUPLE_FREQ_SECS as 300 sec i
我有一个 jsp 代码在其中一台机器上运行良好。但是当我复制到另一台机器时,我得到了这个 no such method found 异常。我是 Spring 的新手。有人可以解释我错过了什么吗? 以下
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
我的代码在下面给出了一个错误; Exception in thread "main" java.lang.NoSuchMethodError: com/myApp/Client.cypherCBC(L
我正在尝试一个 Restful web 服务示例,所以当我要访问 url 时,我遇到了异常 java.lang.NoSuchMethodError: jersey.repackaged.com.goo
我正在将一个 Spring web 项目转换为一个 Maven 项目,但我收到了这个错误: java.lang.NoSuchMethodError: org.jboss.logging.Logger.
在我的项目中,我有一个像这样的枚举: public enum MyEnum { FIRST(1), SECOND(2); private int value; private MyEnum(int v
我创建了这个简单的示例,用于读取 Linux 正常运行时间: public String getMachineUptime() throws IOException { String[] di
我正在使用 Eclipse,并且正在使用 Java。我的目标是使用 bogoSort 方法对 vector 进行排序在一个 vector (vectorExample)中适应我的 vector 类型,
我正在运行以下查询。它显示一条错误消息。如何解决这个错误? ListrouteList=null; List companyList = session.createS
我有以下模型类: @Entity @Table(name="user_content") @org.hibernate.annotations.NamedQueries({ @org.
我有那个错误。这是我的代码: GmailSettingsService service = new GmailSettingsService(APPLICATION_NAME, DOMAIN_NAME
实际上我在执行我的java程序时遇到了下面提到的错误 Exception in thread "pool-1-thread-1" java.lang.ClassCastException: jav
java.lang.ClassCastException: java.lang.Float cannot be cast to java.lang.String 我在以下代码中遇到此异常: Strin
我正在尝试从 linkedhashset 中检索随机元素。下面是我的代码,但它每次都给我异常。 private static void generateRandomUserId(Set userIds
我已经完成了 Android 中的代码: List spinnerArray = new ArrayList(); for (int i = 0; i item = (LinkedTreeMap)
这个问题已经有答案了: Explanation of ClassCastException in Java (12 个回答) 已关闭 6 年前。 我已经编写了 java 到 Json 的代码,同时从页
这个问题在这里已经有了答案: ClassCastException java.lang.Long cannot be cast to clojure.lang.IFn (4 个答案) 关闭 6 年前
我在运行时遇到问题来编译这段代码,这给我一个错误,java.lang.Integer 无法转换为 Java.lang.Double。如果有人帮助我更正此代码,我将非常高兴 double x; pu
我是一名优秀的程序员,十分优秀!