当动态代理遇到ioc

Posted silyvin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了当动态代理遇到ioc相关的知识,希望对你有一定的参考价值。

在 work log 2020.4.28中,我们使用jdk动态代理处理方法的事务增强

 

图1 问题的引出

 

1 起先,我们禁止被增强的类里面使用注解注入依赖,代码会自动审核这种情况

    private static final class TransactionProxyProvider<T> implements Provider<T> {

        private Object target;

        public TransactionProxyProvider(Object target) {
            this.target = target;
        }

        @Override
        public T get() {
            try {
//                Class cl = target.getClass();
//                Field [] fields = cl.getDeclaredFields();
//                for(Field field : fields) {
//                    field.setAccessible(true);
//                    if(field.isAnnotationPresent(Inject.class))
//                        throw new RuntimeException("proxy class do not allow com.google.inject annotation - " + cl.getName());
//                }
                return (T)new TransactionProxyFactory(target).getProxyInstance();
            } catch (Exception e) {
                loggerCommon.error(e.getMessage(), e);
            }
            return null;
        }
    }

  

图2 解决方案

 

public class TransactionProxyFactory implements InvocationHandler {
    private volatile boolean hasNotInjected = true;
    private Object target;
    public TransactionProxyFactory(Object target){
        this.target = target;
    }

    private <T> T getBeanFromFactoryGuice(Class<T> c) {
        Injector injector = CRFGuiceContext.getInjector();
        return injector.getInstance(c);
    }

    private Class getBeanInjectAnnotationGuice() {
        return com.google.inject.Inject.class;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        if(hasNotInjected) {
            synchronized (this) {
                if(hasNotInjected) {
                    Class clProxy = target.getClass();
                    Field [] fields = clProxy.getDeclaredFields();
                    for(Field field : fields) {
                        field.setAccessible(true);
                        if(!field.isAnnotationPresent(getBeanInjectAnnotationGuice()))
                            continue;
                        String name = field.getName();
                        Class fieldType = field.getType();
                        Object obj = getBeanFromFactoryGuice(fieldType);
                        if(obj != null)
                            field.set(target, obj);
                    }
                    hasNotInjected = false;
                }
            }
        }

        SCEF_DB_TRANSACTIONAL scef_db_transactional = method.getAnnotation(SCEF_DB_TRANSACTIONAL.class);
        if(scef_db_transactional != null) {
            ScefOrmSession scefOrmSession = getBeanFromFactoryGuice(ScefOrmSession.class);
            Boolean readOnly = scef_db_transactional.readOnly();
            int isolation = scef_db_transactional.isolation();
            try {
                scefOrmSession.startTransaction(readOnly, isolation);
                Object returnValue = method.invoke(target, args);
                scefOrmSession.commit();
                return returnValue;
            } catch (InvocationTargetException ie) {
                Throwable throwable = ie.getTargetException();
                Class c1 = throwable.getClass();
                Class [] c2 = scef_db_transactional.noRollbackFor();
                int sum = 0;
                for(Class c : c2) {
                    if(c.equals(c1))
                        sum ++ ;
                }
                if(sum == 0)
                    scefOrmSession.rollback();
                else
                    scefOrmSession.commit();
                throw new DBException(throwable);
            } catch (Exception e) {
                throw new DBException(e);
            } finally {
                scefOrmSession.endTransaction();
            }
        } else {
            Object returnValue = method.invoke(target, args);
            return returnValue;
        }

    }

    public Object getProxyInstance(){
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
    }
}

  

以上是关于当动态代理遇到ioc的主要内容,如果未能解决你的问题,请参考以下文章

反射真的慢么?动态代理会创建很多临时class?

2020-07-26 带着疑问看源码 -- springboot aop默认采用啥动态代理机制

Java开发人员必懂的基础——反射与动态代理

Springday02 jabcTemplate注解IOC创建对象注解IOC注入注解 IO生命周期纯注解IOCSpring-junitAOP-JDK和CGLIB动态代理

java的动态代理

动态代理