proxy代理实现接口调用处理

proxy代理实现接口调用处理

我们知道,要调用接口必须要实例化才能调用。
那么我们能不能不实例化调用呢?像mybatis、hibernate那样定义一个接口就能注入调用。其实他们的底层实现就是用代理模式 proxy:
在不实例化的情况下调用接口(实际上代理实例化了)

首先定义一个注解用来区别

import java.lang.annotation.*;

/**
 * @author lingkang
 * Created by 2022/5/4
 * mapper 查询注解
 */
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Select {
    /**
     * 查询语句 SQL
     */
    String value() default "";
}

再编写一个调用接口

import java.util.Map;

/**
 * @author lingkang
 * Created by 2022/5/4
 */
public interface MyMapper {
    @Select("select * from user where id=?")
    Map select(Integer id);
}

然后定义代理调用处理类

import java.io.Serializable;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.HashMap;
import java.util.Map;

/**
 * @author lingkang
 * Created by 2022/5/4
 */
public class MyInvocationHandler implements InvocationHandler, Serializable {
    private Class<?> clazz;

    public MyInvocationHandler(Class<?> clazz) {
        this.clazz = clazz;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Method exMethod = clazz.getMethod(method.getName(), getClassTypes(args));
        if (args[0].getClass().isArray()) {
            args = (Object[]) args[0];
            if (args.length == 0)
                args = null;
        }
        Select select = exMethod.getAnnotation(Select.class);
        System.out.println("查询SQL:" + select.value());
        System.out.println("参数: " + args[0]);

        // 接口需要返回的结果类型
        //Class<?> returnType = exMethod.getReturnType();
        Class<?> returnType = getReturnType(exMethod);
        System.out.println("接口需要返回的结果类型: " + returnType);
        Map map = new HashMap();
        map.put("result", "hello lingkang!");
        System.out.println("假设在此执行SQL返回结果:");
        return map;
    }

    // 有时可能需要特殊处理
    public static Class<?> getReturnType(Method method) {
        Type returnType = method.getGenericReturnType();
        if (returnType instanceof ParameterizedType) {
            ParameterizedType type = (ParameterizedType) returnType;
            return (Class<?>) type.getActualTypeArguments()[0];
        } else if ("void".equals(returnType.getTypeName())) {
            return null;
        }
        return Map.class;
    }

    public static Class<?>[] getClassTypes(Object[] args) {
        Class[] clazz = new Class[args.length];
        for (int i = 0; i < args.length; i++)
            clazz[i] = args[i].getClass();
        return clazz;
    }
}

调用这个代理处理

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

/**
 * @author lingkang
 * Created by 2022/5/4
 */
public class Demo01 {
    public static void main(String[] args) {
        // 获取接口代理
        MyMapper mapper = getMapper(MyMapper.class, new MyInvocationHandler(MyMapper.class));
        // 调用
        System.out.println("调用结果:"+mapper.select(1));
    }

    /**
     * 代理实例化
     */
    public static <T> T getMapper(Class<T> clazz, InvocationHandler handler) {
        return (T) Proxy.newProxyInstance(
                clazz.getClassLoader(),
                new Class[]{clazz},
                handler);
    }
}

结果:
在这里插入图片描述

posted @ 2022-09-16 00:08  凌康  阅读(107)  评论(0编辑  收藏  举报