重要思想 动态代理

模拟某企业用户管理业务,

需包含用户登录,用户删除,用户查询功能,

并要统计每个功能的耗时。

分析 定义一个UserService表示用户业务接口,

规定必须完成用户登录,用户删除,用户查询功能。

定义一个实现类UserServiceImpl实现UserService,并完成相关功能,且统计每个功能的耗时。

定义测试类,创建实现类对象,调用方法。

 

public class Test {
public static void main(String[] args) {
// 1、把业务对象,直接做成一个代理对象返回,代理对象的类型也是 UserService类型
UserService userService = ProxyUtil.getProxy(new UserServiceImpl());

System.out.println(userService.login("admin", "1234"));
System.out.println(userService.deleteUsers());
userService.selectUsers();
userService.updateUsers(); // 走代理
}
}

/**
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
参数一:类加载器,负责加载代理类到内存中使用。
参数二:获取被代理对象实现的全部接口。代理要为全部接口的全部方法进行代理
参数三:代理的核心处理逻辑
*/
public class ProxyUtil {
/**
生成业务对象的代理对象。
* @param obj
* @return
*/
public static <T> T getProxy(T obj) {
// 返回了一个代理对象了
return (T)Proxy.newProxyInstance(obj.getClass().getClassLoader(),
obj.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 参数一:代理对象本身。一般不管
// 参数二:正在被代理的方法
// 参数三:被代理方法,应该传入的参数
long startTimer = System .currentTimeMillis();
// 马上触发方法的真正执行。(触发真正的业务功能)
Object result = method.invoke(obj, args);

long endTimer = System.currentTimeMillis();
System.out.println(method.getName() + "方法耗时:" + (endTimer - startTimer) / 1000.0 + "s");

// 把业务功能方法执行的结果返回给调用者
return result;
}
});
}
}

非常的灵活,支持任意接口类型的实现类对象做代理,

也可以直接为接本身做代理。

可以为被代理对象的所有方法做代理。

可以在不改变方法源码的情况下,实现对方法功能的增强。

不仅简化了编程工作、提高了软件系统的可扩展性,同时也提高了开发效率。

posted on 2022-04-05 15:50  我要当程序源  阅读(16)  评论(0编辑  收藏  举报

导航