Java 模拟Spring,实现IOC和AOP的核心(二)

接着上一篇,在上一篇完成了有关IOC的注解实现,这一篇用XML的方式实现IOC,并且完成AOP。

简易的IOC框图

在这里插入图片描述
注解的方式实现了左边的分支,那么就剩下右边的XML分支:

XmlContext
这个类是也是AbstractApplicationContext的子类,和AnnotationContext相似,只不过这里是要解析XML文件而不是注解:
对于XML文件的处理,是不太容易的,会产生很多问题,后面只是实现核心步骤,很多属性就不考虑了!
首先给出XmlBean,和AnnotationBean一样,都是继承自BeanElement

public class XmlBean implements BeanElement {
    private boolean DI;
    private Object object;
    private Object proxy;
    private Map<Field, String> wiredMap;
    // key:object的为注入成员  value:依赖的className
    // 将不能注入的成员先保存起来

    protected XmlBean() {
        this(true, null, null);
    }

    protected XmlBean(Object object, Object proxy) {
        this(true, object, proxy);
    }

    protected XmlBean(boolean dI, Object object, Object proxy) {
        DI = dI;
        this.object = object;
        this.proxy = proxy;
    }

    protected void addWiredElement(Field field, String ref) throws RepeatProperty {
        if (wiredMap == null) {
            wiredMap = new HashMap<>();
        }
        if (wiredMap.containsKey(field)) {
            throw new RepeatProperty(object.getClass() + "成员:" + field.getName() + "已定义!");
        }
        wiredMap.put(field, ref);
    }

    protected void setDI(boolean DI) {
        this.DI = DI;
    }

    protected Map<Field, String> getWiredMap() {
        return wiredMap;
    }

    @Override
    @SuppressWarnings("unchecked")
    public <E> E getProxy() {
        return (E) proxy;
    }

    @Override
    public Object getObject() {
        return object;
    }

    @Override
    public boolean isDI() {
        return DI;
    }

}

XmlContext

public class XmlContext extends AbstractApplicationContext {
    protected XmlContext() {
    }

    protected XmlContext(String xmlPath) {
        innerParseXml(xmlPath);
    }

    // 和注解方式中的做法一样,只不过产生的是XML方式的BeanElement
    private XmlBean addXmlBean(Class<?> klass, Object object, String classId, String className) throws BeansException {
        Object proxy = aopFactory.creatCGLibProxy(klass, object);
        XmlBean bean = new XmlBean(object, proxy);
        add(classId, className, bean);
        return bean;
    }

    protected void innerParseXml(String xmlPath) {
        // 找到根标签
        new XMLReader() {
            @Override
            public void dealElment(Element element, int index) {
                // 处理bean标签
                new XMLReader() {
                    @Override
                    public void dealElment(Element element, int index) {
                        // 得到id属性和class属性的值
                        String classId = element.getAttribute("id");
                        String className = element.getAttribute("class");
                        try {
                            // 由class得到类
                            Class<?> klass = Class.forName(className);
                            // 处理constructor标签
                            new XMLReader() {
                                @Override
                                public void dealElment(Element element, int index) {
                                // TODO 处理有参数的构造方法,这里就会遇到许多问题,在这里我就不处理了,后面会给出解决思路
                                }
                            }.parse(element, "constructor-arg");
                            // 由于上面没有处理带参数的构造方法,这里直接通过反射机制调用无参构造产生对象
                            // 并且利用产生的对象生成代理对象,最后得到Bean放入beanMap中
                            Object object = klass.newInstance();
                            XmlBean bean = addXmlBean(klass, object, classId, className);

                            // 处理property标签
                            new XMLReader() {
                                @Override
                                public void dealElment(Element element, int index) {
                                    try {
                                        dealProperty(element, klass, bean);
                                    } catch (XmlPropertyMustNeedNameException e) {
                                        e.printStackTrace();
                                    } catch (Exception e) {
                                        e.printStackTrace();
                                    }
                                }
                            }.parse(element, "property");
                        } catch (Exception e1) {
                            e1.printStackTrace();
                        }
                    }
                }.parse(element, "bean");
            }
        }.parse(XMLReader.openDocument(xmlPath), "SimpleSpring");
    }

    private void dealProperty(Element element, Class<?> klass, XmlBean bean)
            throws XmlPropertyMustNeedNameException, Exception {
        // 得到property标签name属性的值
        String fieldName = element.getAttribute("name");
        if (fieldName.length() <= 0) {
            throw new XmlPropertyMustNeedNameException("Bean" + klass.getName() + "的Property标签必须声明name属性!");
        }
        // 通过反射机制得到成员
        Field field = klass.getDeclaredField(fieldName);
        // 得到该成员的类型
        Class<?> fieldType = field.getType();
        // 得到value属性
        String value = element.getAttribute("value");
        // 得到ref属性
        String ref = element.getAttribute("ref");

        // 判断ref和value是否同时存在
        if (value.length() > 0 && ref.length() > 0) {
            throw new CanNotJudgeParameterException("value:" + value + " ref:" + ref + "只能存在一个!");
        }
        Object arg = null;
        // value存在,则直接通过类型转换给成员赋值
        if (value.length() > 0) {
            if (!fieldType.isPrimitive() && !fieldType.equals(String.class)) {
                throw new ValueOnlyPrimitiveType("Value只能用于八大基本类型!");
            }
            // TypeConversion是我自己写的,将字符串转换为基本类型的工具
            arg = TypeConversion.getValue(value, fieldType.getSimpleName());
            field.setAccessible(true);
            field.set(bean.getObject(), arg);
        }
        if (ref.length() > 0) {
            // ref属性存在,由于存在相互依赖关系,所以现在不做处理,只是将其保存起来
            // 设置该bean的状态为尚未注入
            bean.setDI(false);
            bean.addWiredElement(field, ref);
        }
    }

}

XmlContext能做的工作也十分有限,只能完成简单的注入,剩下的注入工作留给下一级处理!

在这里之所以没有处理constructor标签,是因为对与构造方法的处理存在许多因素:
比如:

public class Test {
    public Test(String one, int two) {
        ......
    }
    public Test(int two, String one) {
        ......
    }
}

通过XML文件读取出来的都是字符串,如何区分它是字符串“123”,而不是int类型123?这两个构造方法到底执行哪个?
再比如说:

public Test(int one, int two, Student student) {
        ......
    }

    public Test(String one, int two, Student student) {
        ......
    }

    public Test(int two, String one, Student student) {
        ......
    }

通过反射机制,我们就需要得到构造方法的集合getConstructors();然后筛选出参数个数符合要求的子集,再遍历这个子集的每一个构造方法,然后遍历当前构造方法的所有参数,一个一个比对参数类型是否符合要求,直到找到符合要求的那一个为止,但是,如果说我们是想执行第三个构造方法,它却找到的是第一个,完全就出问题了!

所以Spring的解决办法是给出一个type属性

 <bean id="xxx" class="xxx.xxx.Test">
     <constructor-arg idnex="0" value="1" type="int.class">
     <constructor-arg idnex="1" value="2" type="java.lang.String">
     <constructor-arg idnex="2" ref="student">
 </bean>

只有这样做才能真真区分,所以以后在使用Spring的constructor标签时,当构造方法有歧义时,一定要给出type属性,避免出错,也减少了查找时的遍历!

接下来就是最后一个类,xml分支的最高容器:
ClassPathXmlApplicationContext
上面的XmlContext只是完成了基本的注入问题,还有后续有关于注入之间的依赖关系,甚至是依赖循环

public class ClassPathXmlApplicationContext extends XmlContext {
    public ClassPathXmlApplicationContext() {
    }

    public ClassPathXmlApplicationContext(String xmlPath) {
        super(xmlPath);
    }

    public ClassPathXmlApplicationContext parseXml(String xmlPath) {
        innerParseXml(xmlPath);
        return this;
    }

    @Override
    public <T> T getBean(Class<T> klass) throws BeansException {
        String className = klass.getName();
        BeanElement bean =  beanMap.get(className);

        if (bean == null) {
            throw new BeansException("Bean :" + klass + "不存在!");
        }
        // 在这里还是只考虑XmlBean的注入,不考虑AnnotationBlean注解的完成情况
        if (!bean.isDI() && bean instanceof XmlBean) {
            autowired(className, (XmlBean)bean);
        }

        return bean.getProxy();
    }

    private void autowired(String klassName, XmlBean bean) throws BeansException {
        // 和AnnotationBean的解决思路一样,先设置状态为已注入,防止循环依赖的无限递归
        bean.setDI(true);
        // 得到尚未注入的成员map
        Map<Field, String> wiredMap = bean.getWiredMap();
        if (wiredMap == null || wiredMap.isEmpty()) return;
        // 遍历map
        for (Field field : wiredMap.keySet()) {
            String ref = wiredMap.get(field);
            String tagClassName = beanNameMap.get(ref);
            // ref如果是id则在beanNameMap中找,如果是className就在beanMap中找
            BeanElement wiredBean = tagClassName == null ? beanMap.get(ref) : beanMap.get(tagClassName);
            if (bean == null) {
                return;
            }
            if (!wiredBean.isDI() && wiredBean instanceof XmlBean) {
                autowired(ref, (XmlBean)wiredBean);
            }
            field.setAccessible(true);
            try {
                field.set(bean.getObject(), wiredBean.getObject());
            } catch (Exception e) {
                throw new BeansException(klassName + "依赖关系不正确!");
            }
        }
        wiredMap.clear();
    }

}

看过注解方式的话再看XML就会发现两者其实是一回事,都是通过两者提供的映射关系,利用反射机制完成注入!
只不过两者提供的映射关系在解析起来时各有各的特点!

Xml方式的实现这里就简单实现了,来看看使用情况:

public class StudentA {
    String name;
    private StudentB B;

    public StudentA() {
    }

    @Override
    public String toString() {
        return "A:" + name + "->" +  B;
    }

}

@Component
public class StudentB {
    private String name;
    private StudentC C;

    public StudentB() {
    }

    @Override
    public String toString() {
        return "B:" + name + "->" + C;
    }

}

@Component
public class StudentC {
    private String name;
    private StudentA A;

    public StudentC() {
    }

    @Override
    public String toString() {
        return "C:" + name;
    }

}

xml的配置:

<SimpleSpring>
    <bean id="haha" class="com.zc.ioc.demo.StudentA">
        <property name="name" value="我是A"></property>
        <property name="B" ref="com.zc.ioc.demo.StudentB"></property>
    </bean>
    <bean class="com.zc.ioc.demo.StudentB">
        <property name="name" value="我是B"></property>
        <property name="C" ref="com.zc.ioc.demo.StudentC"></property>
    </bean>
    <bean class="com.zc.ioc.demo.StudentC">
        <property name="name" value="我是C"></property>
        <property name="A" ref="haha"></property>
    </bean>
</SimpleSpring>

主函数:

public static void main(String[] args) throws BeansException {
        // 或者是使用BeanFactory beanFactory = new ClassPathXmlApplicationContext("/test_simple_spring.xml");
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/test_simple_spring.xml");
        StudentA bean = applicationContext.getBean(StudentA.class);
        System.out.println(bean);
}

输出:

在这里插入图片描述

那么试一试注解和Xml方式的混合使用:

@Component
public class StudentA {
    @Value(value="我是A")
    String name;
    @Autowired
    private StudentB B;

    public StudentA() {
    }

    @Override
    public String toString() {
        return "A:" + name + "->" +  B;
    }

}

@Component
public class StudentB {
    @Value(value="我是B")
    private String name;
    @Autowired
    private StudentC C;

    public StudentB() {
    }

    @Override
    public String toString() {
        return "B:" + name + "->" + C;
    }

}
@Component
public class StudentC {
    @Value(value="我是C")
    private String name;
    @Autowired
    private StudentD D;

    @Autowired
    private StudentA A;

    public StudentC() {
    }

    @Override
    public String toString() {
        return "C:" + name + "->" + D;
    }

}

public class StudentD {
    private String name;

    public StudentD() {
    }

    @Override
    public String toString() {
        return "D:" + name;
    }

}

Xml配置:

<SimpleSpring>
     <bean class="com.zc.ioc.demo.StudentD">
         <property name="name" value="我是D"></property>
     </bean>
 </SimpleSpring>

主函数:

public static void main(String[] args) throws BeansException {
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/test_simple_spring.xml");
        StudentD studentD = applicationContext.getBean(StudentD.class);
        System.out.println(studentD);

        applicationContext= new AnnotationConfigApplicationContext("com.zc.moedl");
        StudentA studentA = applicationContext.getBean(StudentA.class);
        System.out.println(studentA);
}

输出结果:

在这里插入图片描述
看起来是没有问题了,但是如果Xml和注解之间的出现顺序不同,结果也会不一样,还得仔细考虑,而且我做的这个是延迟注入,只有在getBean的时候才会完成最后的注入,并且若是注解中需要一个Xml的bean注入,而xml的这个bean又依赖于注解中的一个bean,那么这套方法是不可行的!

AOP

前面多次谈到AOP,以及我们的Bean是通过原始对象+代理对象,这里来看看AOP部分的实现:
AOP说到底主要目的不是产生代理对象,而是要通过代理对象执行方法,并对方法进行有效的拦截!
简单起见,将拦截分为置前,置后,以及出现异常时的拦截。
而拦截又是何时产生的?
还是为了简单实现,后面都只使用CGLibProxy,有关CGLib的代理我在上一篇有介绍,这里也就不累赘了。

关于拦截器的产生,我之前的实现方式是给要拦截的方法添加注解,给出拦截Id,然后提供一套方法,给指定Id号的方法创建拦截器,但是,在知道Spring的处理后,这种方式很快被否定了!在工程中,往往很多需要拦截的方法是不允许侵入式修改的,又或者是被打成了jar包,那么就更不可能对其添加注解,所以给出新的解决思路:
由用户自己写一个方法,然后给这个方法添加注解,使其和要拦截的方法产生对映射关系,这样我们实际执行的拦截器方法完全是由用户提供,并不会干预源代码!
前面说过只是处理置前,置后,以及出现异常时的拦截,所以会给出三种不同的注解,用于区分!
由于是要使用注解,那么就要用到包扫描
包扫描就需要对类进行区分,只处理带有标识的类,所以还缺少一个对类的注解:

@Aspect

 @Retention(RetentionPolicy.RUNTIME)
 @Target(ElementType.TYPE)
 public @interface Aspect {
 }

这个注解只是为了表明这个类存放着用户编写的拦截器方法!

主要的是下面三个注解:

@Before

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Before {
    Class<?> klass();
     String method();
}

置前拦截方法的注解,klass表明对哪个类进行置前拦截,method表明对哪个方法进行拦截,但发现仅仅通过这好像不能找到具体的方法,但仔细想一想,置前拦截是对要拦截的方法参数进行判断,用户在编写拦截时必然知道拦截的方法是什么,参数个数和类型当然也知道,那我们只要让用户写的方法的参数和要拦截的方法参数保持一致就行了,如果不一致,就异常处理!这样就能通过用户编写的方法,知道被拦截的方法参数,进而定位到具体要拦截的方法!

@After

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface After {
    Class<?> klass();
    String method();
    Class<?>[] parameterTypes() default {};
}

置后拦截方法的注解,同置前拦截一样,klass表明对哪个类进行置前拦截,method表明对哪个方法进行拦截。由于之后拦截是对方法执行结果的操作,用户写的方法的参数有且只有一个,且参数类型要与原方法的返回值类型匹配,这样,我们就不能像处理@Before时一样,必须申明被拦截的方法的参数类型,只有这样才能定位到具体的被拦截方法!

@Throwable

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Throwable {
    Class<?> klass();
    String method();
    Class<?>[] parameterTypes();
}

出现异常时拦截的注解,和@After一样,它只是处理异常,那么用户所提供的方法的参数有且只有一个,类型是执行被拦截方法产生的异常类型,它也必须传递被拦截方法的参数,使其定位到具体的被拦截方法!

其实在Spring里面使用了更为犀利的手段,它并没有使用Class<?> 只是使用了一个字符串就解决了类、方法、参数的定位,只不过它就需要对字符串解析,利用了正则表达式,虽然比我的方法繁琐,但面向用户的使用是十分友好的!

AOP框图

在这里插入图片描述

看起来我把它做的很复杂,事实上这里用到的全是接口,是非常灵活的,如果说不想使用这套方式,那么可以自己实现Advice接口;如果说拦截器链用的时list存储,以后想更换为链表也是可以的;拦截器的产生不想使用上面说的注解方式,那么自己去实现IntercepterFactory接口!

AopFactory

public class AopFactory {
    private Advice advice;

    public AopFactory() {
    }

    public AopFactory setAdvice(Advice advice) {
        this.advice = advice;
        return this;
    }

    public <E> E creatCGLibProxy(Class<?> klass) throws Exception {
        return creatCGLibProxy(klass, klass.newInstance());
    }

    public <E> E creatCGLibProxy(Object object) {
        return creatCGLibProxy(object.getClass(), object);
    }

    @SuppressWarnings("unchecked")
    public <E> E creatCGLibProxy(Class<?> klass, Object object) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(klass);
        enhancer.setCallback(new MethodInterceptor() {
            @Override
            public Object intercept(Object proxyObject, Method method,
                    Object[] args, MethodProxy methodProxy) throws Throwable {
                return doInvoke(object, method, args);
            }
        });

        return (E) enhancer.create();
    }

    private Object doInvoke(Object object, Method method,
            Object[] args)
             throws Throwable{
        Object result = null;
        // AdviceAdapter是Advice的适配,什么都不做
        advice = advice == null ? new AdviceAdapter() : advice;
        if (!advice.dealBefore(method, args)) {
            return result;
        }
        try {
            result = method.invoke(object, args);
            result = advice.dealAfter(method, result);
        } catch (Throwable e) {
            advice.delaThrowable(method, e);
            throw e;
        }

        return result;
    }

}

AopFactory只负责产生代理对象,而代理的拦截就下发给Advice

Advice

 public interface Advice {
     boolean dealBefore(Method method, Object[] args);
     Object dealAfter(Method method, Object result);
     void delaThrowable(Method method, Throwable e);
 }

暂时只处理置前、置后、以及异常,以后需要再添加,而不是修改!这就是使用接口的好处!

IntercepterLink

public interface IntercepterLink {
    boolean add(IntercepterMethod tagMethod);
    public boolean doBefore(Object[] args);
    Object doAfter(Object result);
    void doThrowable(Throwable e);
}

拦截器链,一个方法可以由多个拦截器,拦截器链是拦截器方法的真正执行者!提供了添加拦截器,处理置前、置后、异常,也可以给个remove,这里就不写了。

IntercepterFactory

public interface IntercepterFactory {
    void addBeforeIntercepter(Method tagMethod, IntercepterMethod imd);
    void addAfterIntercepter(Method tagMethod, IntercepterMethod imd);
    void addThrowableIntercepter(Method tagMethod, IntercepterMethod imd);

    IntercepterLink getBeforeIntercepterLink(Method tagMethod);
    IntercepterLink getAfterIntercepterLink(Method tagMethod);
    IntercepterLink getThrowableIntercepterLink(Method tagMethod);
}

拦截器链的创建者和拥有着,其中的IntercepterMethod就是拦截器接口:

IntercepterMethod

 public interface IntercepterMethod {
     Object getIntercepterObject();
     Method getIntercepterMethod();
 }

我们的拦截器的执行是通过反射机制,那么就必须知道方法和对象,至于参实是通过CGLib代理机制传递过来的,就不用考虑!

准备工作完成,接下来就是真真的处理部分:
IntercepterFactory在上面说是拦截器的创建者和持有者,所以我把它的是实现类进行了分级:

IntercepterLoader

public class IntercepterLoader implements IntercepterFactory {
    // 可以看到每个方法都有自己的置前、置后、异常拦截器链
    private static  final Map<Method, IntercepterLink> beforeMap;
    private static  final Map<Method, IntercepterLink> afterMap;
    private static  final Map<Method, IntercepterLink> exceptionMap;

    static {
        beforeMap = new HashMap<>();
        afterMap = new HashMap<>();
        exceptionMap = new HashMap<>();
    }

    public IntercepterLoader() {
    }

    @Override
    public IntercepterLink getBeforeIntercepterLink(Method tagMethod) {
        return beforeMap.get(tagMethod);
    }

    @Override
    public IntercepterLink getAfterIntercepterLink(Method tagMethod) {
        return afterMap.get(tagMethod);
    }

    @Override
    public IntercepterLink getThrowableIntercepterLink(Method tagMethod) {
        return exceptionMap.get(tagMethod);
    }

    private void add(Map<Method, IntercepterLink> map,
            Method tagMethod, IntercepterMethod imd) {
        IntercepterLink link = map.get(tagMethod);
        // 防止多线程的访问而创建不同的拦截器链
        if (link == null) {
            synchronized (map) {
                if (link == null) {
                    // IntercepterNodeList是我这套机制默认的IntercepterLink实现类
                    link = new IntercepterNodeList(imd);
                }
            }
            // 该方法还未创建拦截器链
            map.put(tagMethod, link);
        } else {
            // 方法相同,则在拦截器链上追加
            link.add(imd);
        }
    }

    @Override
    public void addBeforeIntercepter(Method tagMethod, IntercepterMethod imd) {
        add(beforeMap, tagMethod, imd);
    }

    @Override
    public void addAfterIntercepter(Method tagMethod, IntercepterMethod imd) {
        add(afterMap, tagMethod, imd);
    }

    @Override
    public void addThrowableIntercepter(Method tagMethod, IntercepterMethod imd) {
        add(exceptionMap, tagMethod, imd);
    }

}

真正意义上的拦截器持有者,它要完成的功能非常简单!

我们是要通过注解的方式产生拦截器,所以就有更高级来处理:

IntercepterLoaderFactory

/**
*  使用包扫描,找到带有@Before、@After、@Throwable的方法,将其添加至拦截器map中
*/
public class IntercepterLoaderFactory extends IntercepterLoader {
    public IntercepterLoaderFactory() {
    }

    public IntercepterLoaderFactory parseMethodForPackage(String packageName) {
        new PackageScanner() {
            @Override
            public void dealClass(Class<?> klass) {
                // 判断类是否满足我们定义的@Aspect
                if (!klass.isAnnotationPresent(Aspect.class)) return;
                try {
                    // 产生方法执行的对象
                    Object object = klass.newInstance();
                    Method[] methods = klass.getDeclaredMethods();
                    // 遍历所有方法,处理带有注解的方法
                    for (Method method : methods) {
                        if (method.isAnnotationPresent(Before.class)) {
                            parseBeforeIntercepter(klass, object, method, method.getAnnotation(Before.class));
                        } else if (method.isAnnotationPresent(After.class)) {
                            parseAfterIntercepter(klass, object, method, method.getAnnotation(After.class));
                        } else if (method.isAnnotationPresent(Throwable.class)) {
                            parseExceptionIntercepter(klass, object, method, method.getAnnotation(Throwable.class));
                        }
                    }
                } catch (InstantiationException e) {
                    e.printStackTrace();
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }
            }
        }.scanPackage(packageName);
        return this;
    }

    /**
    *    处理带@Before注解的方法
    */
    private void parseBeforeIntercepter(Class<?> klass, Object object, Method method, Before before) {
        if (!method.getReturnType().equals(boolean.class)) {
            try {
                throw new ReturnTypeNotMatch(method + "返回值类型必须是boolean!");
            } catch (ReturnTypeNotMatch e) {
                e.printStackTrace();
            }
        }
        // 从@Before注解中获取被拦截方法的信息
        Class<?> targetClass = before.klass();
        String targetMethodName = before.method();
        Class<?>[] methodTypes = method.getParameterTypes();

        try {
            Method targetMethod = targetClass.getDeclaredMethod(targetMethodName, methodTypes);
            // 父类的方法调用,其中的IntercepterMethodDefination是IntercepterMethod的实现类
            addBeforeIntercepter(targetMethod, new IntercepterMethodDefination(object, method));
        } catch (NoSuchMethodException e) {
            try {
                throw new IntercepterMethodParaNotMatch(method + "参数不匹配!");
            } catch (IntercepterMethodParaNotMatch e1) {
                e1.printStackTrace();
            }
        }
    }

    /**
    *    处理带@After注解的方法
    */
    private void parseAfterIntercepter(Class<?> klass, Object object, Method method, After after) {
        //    从@After注解中获取被拦截方法的信息
        Class<?> targetClass = after.klass();
        String targetMethodName = after.method();
        Class<?>[] targetMethodPara = after.parameterTypes();
        try {
            // 通过上述参数得到被拦截方法,如果得不到,异常处理
            Method targetMethod = targetClass.getDeclaredMethod(targetMethodName, targetMethodPara);
            Class<?> targetMethodReturnType = targetMethod.getReturnType();
            Class<?> methodReturnType = method.getReturnType();
            Class<?>[] methodParameters = method.getParameterTypes();
            //    判断是否满足置后拦截方法的条件:
            //    置后拦截的方法返回值类型必须和被拦截方法相同
            //    置后拦截的方法的参数有且只有一个,且是被拦截的方法返回值类型
            if (methodParameters.length != 1
                    || !targetMethodReturnType.equals(methodReturnType)
                    || !methodReturnType.equals(methodParameters[0])) {
                try {
                    throw new IntercepterMethodParaNotMatch("拦截器方法:" + method +
                            " 与被拦截方法" + targetMethod + "不匹配!");
                } catch (IntercepterMethodParaNotMatch e) {
                    e.printStackTrace();
                }
            }

            addAfterIntercepter(targetMethod, new IntercepterMethodDefination(object, method));
        } catch (NoSuchMethodException | SecurityException e) {
            try {
                throw new IntercepterMethodParaNotMatch("被拦截方法[" + targetMethodName + "]不存在!");
            } catch (IntercepterMethodParaNotMatch e1) {
                e1.printStackTrace();
            }
        }
    }

    private void parseExceptionIntercepter(Class<?> klass, Object object, Method method, Throwable throwable) {
            //    从@Throwable 注解中获取被拦截方法的信息
            Class<?> targetClass = throwable.klass();
            String targetMethodName = throwable.method();
            Class<?>[] targetMethodPara = throwable.parameterTypes();
            try {
                // 通过上述参数得到被拦截方法,如果得不到,异常处理
                Method targetMethod = targetClass.getDeclaredMethod(targetMethodName, targetMethodPara);
                Class<?>[] methodParameters = method.getParameterTypes();
                //    判断是否满足异常拦截方法的条件:
                //    异常拦截的方法的参数有且只有一个,且是java.lang.Throwable
                if (methodParameters.length != 1
                        || methodParameters[0].equals(java.lang.Throwable.class)) {
                    try {
                        throw new IntercepterMethodParaNotMatch("拦截器方法:" + method +
                                " 与被拦截方法" + targetMethod + "不匹配!");
                    } catch (IntercepterMethodParaNotMatch e) {
                        e.printStackTrace();
                    }
                }
                addAfterIntercepter(targetMethod, new IntercepterMethodDefination(object, method));
            } catch (NoSuchMethodException | SecurityException e) {
                try {
                    throw new IntercepterMethodParaNotMatch("被拦截方法[" + targetMethodName + "]不存在!");
                } catch (IntercepterMethodParaNotMatch e1) {
                    e1.printStackTrace();
                }
            }
    }

}

通过这套机制,我们就能通过注解+包扫描,十分方便地给指定方法添加拦截了!

IntercepterMethodDefination

public class IntercepterMethodDefination implements IntercepterMethod {
    private Object intercepterObject;
    private Method intercepterMethod;

    protected IntercepterMethodDefination() {
    }

    protected IntercepterMethodDefination(Object intercepterObject, Method intercepterMethod) {
        this.intercepterObject = intercepterObject;
        this.intercepterMethod = intercepterMethod;
    }

    @Override
    public Object getIntercepterObject() {
        return intercepterObject;
    }

    @Override
    public Method getIntercepterMethod() {
        return intercepterMethod;
    }

}

拦截器方法执行所需的封装

拦截器我们也有了,就剩下拦截器链了:
我的拦截器链使用了链表,为了能够方法地链式调用,也就是设计模式之一的职责链模式,当然也可以使用List,只不过使用链表相比于List,在处理时都需要遍历,没有什么差别,但是链表比List占的空间小,List在内部是数组,且数组大小是大于有效元素个数的!

IntercepterNodeList

public class IntercepterNodeList implements IntercepterLink {
    private IntercepterMethod imd; // 拦截器
    private IntercepterNodeList next; // 下一结点
    private IntercepterNodeList last; // 尾结点

    protected IntercepterNodeList() {
        this(null);
    }

    protected IntercepterNodeList(IntercepterMethod imd) {
        this.imd = imd;
        this.next = null;
        this.last = this;
    }

    /**
    *    尾插法追加结点
    */
    @Override
    public boolean add(IntercepterMethod imd) {
        if (next == null) {
            next = new IntercepterNodeList(imd);
            last = next;
        } else {
            last = last.next = new IntercepterNodeList(imd);
            last.next = null;
        }
        return true;
    }
    /**
    *    链式调用处理置前拦截
    */
    @Override
    public boolean doBefore(Object[] args) {
        boolean isContinue = this.innerInvoke(imd.getIntercepterObject(), imd.getIntercepterMethod(), args);
        if (this.next != null && isContinue) {
            isContinue = this.next.doBefore(args);
        }
        return true;
    }

    @SuppressWarnings("unchecked")
    private <T> T innerInvoke(Object object, Method method, Object[] args) {
        T result = null;
        try {
            result = (T) method.invoke(object, args);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return result;
    }

    /**
    *    链式调用处理置后拦截
    */
    @Override
    public Object doAfter(Object result) {
        result = innerInvoke(imd.getIntercepterObject(), imd.getIntercepterMethod(), new Object[] {result});
        if (this.next != null) {
            result = this.next.doAfter(result);
        }
        return result;
    }

    /**
    *    链式调用处理异常拦截
    */
    @Override
    public void doThrowable(Throwable e) {
        innerInvoke(imd.getIntercepterObject(), imd.getIntercepterMethod(), new Object[] {e});
        if (this.next != null) {
            this.next.doThrowable(e);
        }
    }

}

AOP到这里就已经结束了,来看看它的使用吧:
被拦截类及其方法:

public class Test {
    public Test() {
    }

    public String fun(int arg) {
        System.out.println("Test的fun方法执行 arg = " + arg);
        return "fun";
    }
}

拦截器所在类:

@Aspect
public class Action {
    public Action() {
    }

    @Before(klass=Test.class, method="fun")
    public boolean beforeFun(int arg) {
        System.out.println("置前拦截beforeFun:arg = " + arg);
        return true;
    }

    @Before(klass=Test.class, method="fun")
    public boolean beforeFunOther(int arg) {
        System.out.println("置前拦截beforeFunOther:arg = " + arg);
        return true;
    }

    @After(klass=Test.class, method="fun", parameterTypes= {int.class})
    public String AfterFun(String arg) {
        System.out.println("置后拦截:arg = " + arg);

        return "AfterFun";
    }
}

主函数:

public static void main(String[] args) throws Exception {
        IntercepterLoaderFactory intercepterLoaderFactory =
                new IntercepterLoaderFactory().parseMethodForPackage("com.zc.action");

        AopFactory aopFactory = new AopFactory();
        aopFactory.setAdvice(new AdviceHander()
                .setIntercepterFactory(intercepterLoaderFactory));

        Test testProxy = aopFactory.creatCGLibProxy(Test.class);
        System.out.println(testProxy.fun(10));
}

执行结果:

在这里插入图片描述

这样的用法是有些恶心了,但是,别忘了,AOP配合IOC才是使用的精华:
注解方式的注入:

@Component
public class StudentA {
    @Value(value="我是A")
    String name;
    @Autowired
    private StudentB B;

    public String fun(int arg) {
        System.out.println("StudentA的fun方法执行 arg = " + arg);
        return "fun";
    }

    @Override
    public String toString() {
        return "A:" + name + "->" +  B;
    }

}

@Component
public class StudentB {
    @Value(value="我是B")
    private String name;
    @Autowired
    private StudentC C;

    public StudentB() {
    }

    @Override
    public String toString() {
        return "B:" + name + "->" + C;
    }

}

@Component
public class StudentC {
    @Value(value="我是C")
    private String name;
    @Autowired
    private StudentD D;

    @Autowired
    private StudentA A;

    public StudentC() {
    }

    @Override
    public String toString() {
        return "C:" + name + "->" + D;
    }

}

Xml方式的注入:

 <SimpleSpring>
     <bean class="com.zc.ioc.demo.StudentD">
         <property name="name" value="我是D"></property>
     </bean>
 </SimpleSpring>

拦截器:

@Aspect
public class Action {
    public Action() {
    }

    @Before(klass=StudentA.class, method="fun")
    public boolean beforeFun(int arg) {
        System.out.println("置前拦截beforeFun:arg = " + arg);
        return true;
    }

    @Before(klass=StudentA.class, method="fun")
    public boolean beforeFunOther(int arg) {
        System.out.println("置前拦截beforeFunOther:arg = " + arg);
        return true;
    }

    @After(klass=StudentA.class, method="fun", parameterTypes= {int.class})
    public String AfterFun(String arg) {
        System.out.println("置后拦截:arg = " + arg);
        return "AfterFun";
    }

}

主函数:

public static void main(String[] args) throws Exception {
        new IntercepterLoaderFactory().parseMethodForPackage("com.zc.action");
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/test_simple_spring.xml");
        StudentD studentD = applicationContext.getBean(StudentD.class);
        System.out.println(studentD);

        applicationContext = new AnnotationConfigApplicationContext("com.zc.model");
        StudentA studentA = applicationContext.getBean(StudentA.class);
        studentA.fun(10);
        System.out.println(studentA);
}

执行结果:

在这里插入图片描述

Spring的IOC和AOP就是先到这里了,欢迎评论区讨论。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/481318.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

海量数据处理项目-技术Leader必备方法论-SMART衡量需求-工作的利器

海量数据处理项目-技术Leader必备方法论-SMART衡量需求-工作的利器

阿里的库存秒杀是如何实现的?

一、阿里的库存秒杀的实现 阿里有很多业务&#xff0c;几十上百个业务线&#xff0c;各自都有一些需要做抢购、秒杀、热点扣将的场景。他们都用哪些方案呢? 我看了很多资料&#xff0c;也找了很多人做交流&#xff0c;最终得到的结论是啥都有&#xff0c;主要总结几个主流的&…

STM32 ESP8266模块的曲折探索

这是本文的配套资料&#xff0c;最终工程请参考 新_ESP8266资料\stm32f103成功移植的项目 【免费】stm32f103c8t6esp8266资料资源-CSDN文库 一、等到了ready 产品参数 我使用的是ai-thinker的esp8266-01s&#xff0c;以下为产品规格书 引脚定义&#xff1a; 依据引脚定义&…

【深度学习目标检测】二十四、基于深度学习的疲劳驾驶检测系统-含数据集、GUI和源码(python,yolov8)

设计一个疲劳驾驶检测系统的重要性主要体现在以下几个方面&#xff1a; 提高道路安全&#xff1a;疲劳驾驶是引发交通事故的重要因素之一。驾驶员在长时间驾驶或缺乏休息的情况下&#xff0c;反应速度和判断能力会显著下降&#xff0c;从而增加事故风险。通过实时检测驾驶员的疲…

金三银四面试题(一):JVM类加载与垃圾回收

面试过程中最经典的一题&#xff1a; 请你讲讲在JVM中类的加载过程以及垃圾回收&#xff1f; 加载过程 当Java虚拟机&#xff08;JVM&#xff09;启动时&#xff0c;它会通过类加载器&#xff08;ClassLoader&#xff09;加载Java类到内存中。类加载是Java程序运行的重要组成…

AI视频风格转换动漫风:Stable Diffusion+TemporalKit

话不多说&#xff0c;直接开干。 基本方法 首先通过 Temporal-Kit 这个插件提取视频中的关键帧图片&#xff0c;然后使用 Stable Diffusion WebUI 重绘关键帧图片&#xff0c;然后再使用 Temporal-Kit 处理转换后的关键帧图片&#xff0c;它会自动补充关键帧之间的图片&#…

Karmada 管理有状态应用 Xline 的早期探索与实践

背景与动机 目前随着云原生技术和云市场的不断成熟&#xff0c;越来越多的 IT 厂商开始投入到跨云多集群的怀抱当中。以下是 flexera 在 2023 年中关于云原生市场对多云多集群管理的接受程度的调查报告&#xff08;http://info.flexera.com&#xff09; 从 flexera 的报告中可…

优化选址问题 | 基于灰狼算法求解基站选址问题含Matlab源码

目录 问题代码问题 灰狼优化算法(Grey Wolf Optimizer, GWO)是一种基于自然界中灰狼群体狩猎行为的优化算法。这种算法通过模拟灰狼的社会等级和狩猎行为来寻找问题的最优解。 基站选址问题通常是一个多目标优化问题,涉及到覆盖范围、信号质量、成本等多个因素。使用灰狼算…

拌合楼管理软件开发(十一) 海康威视车牌识别摄像头安装调试,记录犯经验主义错误不断自己打脸过程

前言: 从小白开始 海康威视的摄像头接触过,包括前面也都开发了调用sdk开发拍照和视频预览,以及通过事件警报获取数据的。接触到的像头都是12v或者24v电源&#xff0c;或者是POE供电的&#xff0c;先入为主了觉得都是这样&#xff0c;结果打脸了。 一、设备选型&#xff1a; 最开…

glibc内存管理ptmalloc - largebin

前言 上节《glibc内存管理ptmalloc》我们讲了fastbin/unsortedbin/smallbin, 有意避开了largebin, 因为largebin稍微复杂一点点&#xff0c;需要单独一节讲解。 largebin的特点 chunk size 大于等于1024字节largebin共有63个链表largebin每个链表中的chunk size不固定largeb…

阿里云2核4G云服务器165元一年,ECS u1优惠价格199元一年

阿里云2核4G服务器租用优惠价格&#xff0c;轻量2核4G服务器165元一年、u1服务器2核4G5M带宽199元一年、云服务器e实例30元3个月&#xff0c;活动链接 aliyunfuwuqi.com/go/aliyun 活动链接如下图&#xff1a; 阿里云2核4G服务器优惠价格 轻量应用服务器2核2G4M带宽、60GB高效…

MySQL数据库 - 存储引擎

一. mysql 存储引擎的相关知识 1.1 存储引擎的概念 MySQL中的数据用各种不下同的技术存储在文件中&#xff0c;每一种技术都使用不同的存储机制、索引技巧、锁定水平并最终提供不同的功能和能力&#xff0c;这些不同的技术以及配套的功能在MySQL中称为存储引擎。存储引擎是My…

天水麻辣烫火爆出圈,秦安能否接得住这“泼天富贵”?

文章目录 泼天富贵来袭莲花社火助力秦安花椒 秦安介绍基本概况秦安地形秦安气候秦安乡镇 秦安旅游李元芳故居兴国寺女娲庙秦安文庙泰山庙人民街古建商业一条街大地湾遗址三国街亭古战场遗址女娲洞 泼天富贵来袭 一碗麻辣烫带火一座城。甘肃天水麻辣烫在社交媒体平台火爆出圈&a…

算法系列--动态规划--回文子串系列

&#x1f495;"我们好像在池塘的水底&#xff0c;从一个月亮走向另一个月亮。。"&#x1f495; 作者&#xff1a;Mylvzi 文章主要内容&#xff1a;算法系列–动态规划–回文子串系列 今天为大家带来的是算法系列--动态规划--回文子串系列(1),本文重点掌握如何快速判…

Linux相关命令(2)

1、W &#xff1a;主要是查看当前登录的用户 在上面这个截图里面呢&#xff0c; 第一列 user &#xff0c;代表登录的用户&#xff0c; 第二列&#xff0c; tty 代表用户登录的终端号&#xff0c;因为在 linux 中并不是只有一个终端的&#xff0c; pts/2 代表是图形界面的第…

伦敦银操作建议中所蕴含的支撑阻力位技术

在伦敦银操作建议或者报告中&#xff0c;尤其是有关伦敦银操作技术分析的建议中&#xff0c;我们总是能看到一个名词&#xff1a;支撑阻力位。其实支撑阻力位有两个意思&#xff0c;一个是支撑位&#xff0c;一个是阻力位&#xff0c;我们习惯将它们合起来称呼&#xff0c;实际…

uniapp安装axios

先npm安装 npm i axios然后在项目里面建一个utils文件&#xff0c;再建一个index.js 以下是index.js代码&#xff1a; import axios from axios; const service axios.create({baseURL: //xxxx.xxxxx.com///你的请求接口域名, timeout: 6000, // request timeoutcrossDomai…

设计数据库之外部模式:数据库的应用

Chapter5&#xff1a;设计数据库之外部模式&#xff1a;数据库的应用 笔记来源&#xff1a;《漫画数据库》—科学出版社 设计数据库的步骤&#xff1a; 概念模式 概念模式(conceptual schema)是指将现实世界模型化的阶段进而&#xff0c;是确定数据库理论结构的阶段。 概念模…

【 yolo红外微小无人机-直升机-飞机-飞鸟目标检测】

yolo无人机-直升机-飞机-飞鸟目标检测 1. 小型旋翼无人机目标检测2. yolo红外微小无人机-直升机-飞机-飞鸟目标检测3. yolo细分类型飞机-鸟类-无人机检测4. yolo红外大尺度无人机检测 1. 小型旋翼无人机目标检测 类别 nc: 1 names: [‘drone’] 数据集和模型 VOC无人机UAV数据…

小目标检测常见解决策略总结

1. 引言 尽管目标检测算法取得了长足的发展&#xff0c;例如 Faster RCNN、YOLO、SSD、RetinaNet、EfficientDet 等。通常&#xff0c;这些模型是在 COCO数据集上训练的。它是一个包含各种对象类别和标注的大规模数据集&#xff0c;因此在训练对象检测器方面很受欢迎。然而&am…