Spring Aop ProxyFactoryBean和ProxyFactory以及ClassCastException

问题描述 投票:0回答:2

当我使用ProxyFactoryBean获取代理对象时,我得到一个ClassCastException,但是当我使用ProxyFactory的getProxy()来获取代理对象时,它可以正常工作。我使用Spring 4.x.

两个豆,WaiterTest和卖方的定义:

public class WaiterTest {
    public void greetTo(String name){
        System.out.println("waiter greet to " + name +"...");
    }
    public void serveTo(String name){
        System.out.println("waiter serving " + name + "...");
    }
}

public class Seller {
    public void greetTo(String name){
        System.out.println("seller greet to " +name + "...");
    }
}

建议的定义:

public class GreetingBeforeAdvice implements MethodBeforeAdvice{

    public void before(Method method, Object[] args, Object obj) throws Throwable{
        System.out.println(obj.getClass().getName() + "." + method.getName());
        String clientName =(String) args[0];
        System.out.println("How are you! Mr." + clientName +".");
    }
}

顾问的定义:

public class GreetingAdvisor extends StaticMethodMatcherPointcutAdvisor {
    public boolean matches(Method method, Class clazz) {
        return "greetTo".equals(method.getName());
    }

    public ClassFilter getClassFilter() {
        return new ClassFilter() {
            public boolean matches(Class clazz) {
                return WaiterTest.class.isAssignableFrom(clazz);
            }
        };
    }
}

测试类:

   public class TestGreetingBeforeAdvisor {
        public static void main(String[] args) {
            //method one:  use by ProxyFactory
            WaiterTest targetWaiterTest = new WaiterTest();
            Seller targetSeller = new Seller();

            GreetingBeforeAdvice advice = new GreetingBeforeAdvice();
            GreetingAdvisor advisor = new GreetingAdvisor();
            advisor.setAdvice(advice);

            ProxyFactory pf = new ProxyFactory();
            pf.setTarget(targetWaiterTest);
            pf.addAdvisor(advisor);
            pf.setOptimize(true);
            WaiterTest proxy = (WaiterTest) pf.getProxy();
            proxy.greetTo("John");
            proxy.serveTo("Tom");

            ProxyFactory pf1 = new ProxyFactory();
            pf1.setTarget(targetSeller);
            pf1.addAdvisor(advisor);
            Seller seller = (Seller) pf1.getProxy();
            seller.greetTo("John"); 
            System.out.println("=============");


            //method two:Spring xml,use ProxyFactoryBean
            ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
                 WaiterTest test = (WaiterTest) ctx.getBean("waiterTest");
        test .greetTo("John");
        test .serveTo("John");
    }
 }

beans.xml中:

<bean id="waiterTarget" class="com.xxx.springaop.advisor.WaiterTest"/>
    <bean id="sellerTarget" class="com.xxx.springaop.advisor.Seller"/>
    <bean id="greetingAdvice1" class="com.xxx.springaop.advisor.GreetingBeforeAdvice"/>
    <bean id="greetingAdvisor" class="com.xxx.springaop.advisor.GreetingAdvisor"
        p:advice-ref="greetingAdvice1"/>
    <bean id="parent" abstract="true" class="org.springframework.aop.framework.ProxyFactoryBean"
        p:interceptorNames="greetingAdvisor" 
        p:proxyTargetClass="true"/>

    <bean id="waiterTest" parent="parent" p:target-ref="waiterTarget"/>
    <bean id="seller" parent="parent" p:target-ref="sellerTarget"/>

结果:

com.xxx.springaop.advisor.WaiterTest.greetTo
How are you! Mr.John.
waiter greet to John...
waiter serving Tom...
seller greet to John...
=============
Exception in thread "main" java.lang.ClassCastException: com.sun.proxy.$Proxy12 cannot be cast to com.xxx.springaop.advisor.WaiterTest
    at com.xxx.springaop.advisor.TestGreetingBeforeAdvisor.main(TestGreetingBeforeAdvisor.java:48)

总结:

WaiterTest proxy = (WaiterTest) pf.getProxy(); //success
WaiterTest test = (WaiterTest) ctx.getBean("waiterTest");//fail,ClassCastException

为什么?

spring spring-aop proxyfactory
2个回答
0
投票

我无法重现您的问题,但我对您的代码进行了一些重构,这对我来说效果很好。也许它至少可以帮助你作为起点..

waiter.Java

public class Waiter {
    public void greetTo(String name) {
        System.out.println("waiter greet to " + name + "...");
    }

    public void serveTo(String name) {
        System.out.println("waiter serving " + name + "...");
    }
}

greeting before advice.Java

import org.springframework.aop.MethodBeforeAdvice;

import java.lang.reflect.Method;

public class GreetingBeforeAdvice implements MethodBeforeAdvice {

    @Override
    public void before(Method method, Object[] args, Object obj) throws Throwable {
        String clientName = (String) args[0];
        System.out.println("How are you! Mr." + clientName + ".");
    }
}

greeting advisor.Java

import org.springframework.aop.ClassFilter;
import org.springframework.aop.support.StaticMethodMatcherPointcutAdvisor;

import java.lang.reflect.Method;

public class GreetingAdvisor extends StaticMethodMatcherPointcutAdvisor {

    public boolean matches(Method method, Class clazz) {
        return "greetTo".equals(method.getName());
    }

    public ClassFilter getClassFilter() {
        return new ClassFilter() {
            public boolean matches(Class clazz) {
                return Waiter.class.isAssignableFrom(clazz);
            }
        };
    }
}

beans.xml中

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
                           http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean id="waiter" class="Waiter"/>

    <!-- Advisor Configuration -->
    <bean id="greetingAdvice" class="GreetingBeforeAdvice"/>
    <bean id="greetingAdvisor" class="GreetingAdvisor">
        <property name="advice" ref="greetingAdvice"/>
    </bean>

    <bean id="waiterProxyFactory" class="org.springframework.aop.framework.ProxyFactoryBean">
        <property name="interceptorNames">
            <list>
                <value>greetingAdvisor</value>
            </list>
        </property>
        <property name="target" ref="waiter"/>
    </bean>
</beans>

最后是App.java

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class App {

    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
        final Waiter waiter = (Waiter) ctx.getBean("waiterProxyFactory");
        waiter.greetTo("Koray Tugay");
        waiter.serveTo("Koray Tugay");
    }
}

使用Spring 4.3.12,我的输出如下:

How are you! Mr.Koray Tugay.
waiter greet to Koray Tugay...
waiter serving Koray Tugay...

0
投票

我会说你没有在ProxyFactoryBean定义中配置'target'对象。如果没有目标,则无法创建代理。

© www.soinside.com 2019 - 2024. All rights reserved.