zoukankan      html  css  js  c++  java
  • Spring mvc 中有关 Shiro 1.2.3 配置问题

    Spring 版本:3.2.x,  4.0.x

    【问题说明】

    首先介绍下配置出错情况:

    (1)项目中,Spring3 and Spring4 的 applicationContext.xml aop 配置如下:

    ... ...
       <aop:aspectj-autoproxy expose-proxy="true"/>    
       <tx:annotation-driven transaction-manager="transactionManager"/>
       <tx:advice id="txAdvice" transaction-manager="transactionManager">
            <tx:attributes>
                <tx:method name="save*" propagation="REQUIRED" />
                <tx:method name="add*" propagation="REQUIRED" />
                <tx:method name="create*" propagation="REQUIRED" />
                <tx:method name="insert*" propagation="REQUIRED" />
                <tx:method name="update*" propagation="REQUIRED" />
                <tx:method name="del*" propagation="REQUIRED" />
                <tx:method name="get*" propagation="REQUIRED" read-only="true" />
                <tx:method name="find*" propagation="REQUIRED" read-only="true" />
                <tx:method name="query*" propagation="REQUIRED" read-only="true" />
                <tx:method name="*"  read-only="true"/>
            </tx:attributes>
        </tx:advice>
       
        <aop:config expose-proxy="true">
            <!-- 只对业务逻辑层实施事务 -->
            <aop:pointcut id="txPointcut" expression="execution(* com.app..service..*.*(..))" />
            <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
        </aop:config>
    ... ...

     未加入 Shiro 之前,所有 applicationContext 的 getBean(...) 方法能正常获取到相应对象实例。 调用 getBeanDefinitionNames() 输出的 Beans 如下:

    Bean Name Bean Class
    userDaoImpl class com.app.dao.impl.UserDaoImpl
    resourceServiceImpl class com.app.service.impl.ResourceServiceImpl$$EnhancerBySpringCGLIB$$4ddf6901
    roleServiceImpl class com.app.service.impl.RoleServiceImpl
    userServiceImpl class com.app.service.impl.UserServiceImpl$$EnhancerBySpringCGLIB$$e7bcd944

    以上 bean 都采用 @Repository 或 @Service 的注解方式。

    注意标红Class名称,这是因为被 CGLIB 动态代理。 

    (2)根据 Shiro 示例,新增配置文件 applicationContext-security.xml,然后配置 Shiro 如下:

    ... ...
    
        <bean id="securityManager" class="org.apache.shiro.web.mgt.DefaultWebSecurityManager">
            <property name="realm" ref="myRealm"/>
        </bean>
    
        <bean id="lifecycleBeanPostProcessor" class="org.apache.shiro.spring.LifecycleBeanPostProcessor"/>
    
        <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" depends-on="lifecycleBeanPostProcessor">
        </bean>
    
        <bean class="org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor">
            <property name="securityManager" ref="securityManager"/>
        </bean>
         
        <bean id="formAuthenticationFilter"  class="org.apache.shiro.web.filter.authc.FormAuthenticationFilter"/>
    
    ... ...

     加入 Shiro 之后,发现有些地方 getBean(...) 出错,调用 getBeanDefinitionNames() 输出的 Beans 如下:

    Bean Name Bean Class
    userDaoImpl class com.app.dao.impl.UserDaoImpl
    resourceServiceImpl class $Proxy38
    roleServiceImpl class com.app.service.impl.RoleServiceImpl
    userServiceImpl class $Proxy32

    可以注意到标红Class名称,这是由于配置 DefaultAdvisorAutoProxyCreator,又被 JDK 动态二次代理。 

    【解决办法】 

    两次都以 CGLIB 动态代理。在 AOP 配置加上 proxy-target-class 属性,值为 true。具体如下:

    (1)在 applicationContext.xml 中配置

    ... ...
        <aop:config expose-proxy="true" proxy-target-class="true">
            <!-- 只对业务逻辑层实施事务 -->
            <aop:pointcut id="txPointcut" expression="execution(* com.app..service..*.*(..))" />
            <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
        </aop:config>
    ... ...

    (2)在 applicationContext-security.xml 配置 Shiro

    ... ...
        <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" depends-on="lifecycleBeanPostProcessor">
            <property name="proxyTargetClass" value="true"/>  
        </bean>
    ... ...

    修改完成后,重新运行程序,getBean 恢复正常。

    【参考资料】

    spring的二次代理原因及如何排查  http://jinnianshilongnian.iteye.com/blog/1894465

  • 相关阅读:
    无监督学习
    监督学习
    cmd
    oj1026
    oj1025
    使用虚函数的不同模式
    hdu1166:敌兵布阵(树状数组或线段树)
    传纸条(动态规划)
    SDUT 1266 出栈序列统计(卡特兰数)
    HDU 5063 Operation the Sequence
  • 原文地址:https://www.cnblogs.com/FrankTang/p/springmvc_shiro_dynamicproxy.html
Copyright © 2011-2022 走看看