动态代理: 顾名思义代理对象的类是经过动态方式来自动生成的。这样的好处是, 咱们不须要每次为被代理对象单首创建代理类, JDK API 中, 对动态代理模式提供了支持。java
JDK 的动态代理支持 :git
java.lang.reflect.Proxy:生成动态代理类和对象;
github
java.lang.reflect.InvocationHandler(处理器接口):能够经过invoke方法实现对真实角色的代理访问。每次经过 Proxy 生成的代理类对象都要指定对应的处理器对象。
数据库
CGLIB实现动态代理 :缓存
CGLIB是高性能的代码生成包。 能够为没有实现接口的类提供代理, 是JDK动态代理必需要有接口的补充。ide
一般能够使用Java的动态代理建立代理, 但当要代理的类没有实现接口或者为了更好的性能, CGLIB是一个好的选择。性能
CGLIB做为一个开源项目, 代码托管在github, 地址为: https://github.com/cglib/cglib
举例说明:操做员查询销售订单功能, 为了提升查询性能, 咱们接入缓存功能, 若是缓存中有数据, 那么直接返回缓存中的数据,;若是缓存中没有数据, 那么查询订单数据后, 把数据加入到缓存, 这样在下次查询订单数据时, 就能够从缓存中读取了。
测试
a.建立订单类、模拟的数据库类、模拟的缓存类;this
//模拟的数据库类:提供getOrder方法来经过oid查询订单信息
public class DB {
private static List<Order> list = new ArrayList<>();
static{
list.add(new Order(1,"毛巾3条",45,"2020-2-1"));
list.add(new Order(2,"小纸巾10包",12,"2020-1-21"));
list.add(new Order(3,"洗发水1瓶",32,"2020-1-30"));
list.add(new Order(4,"红牛1箱",36,"2020-2-2"));
list.add(new Order(5,"哈脾2箱",120,"2020-2-3"));
}
public Order getOrder(int oid){
System.out.println("从数据库中查找数据...");
for(Order order:list){
if(order.getOid() == oid){
return order;
}
}
return null;
}
}
//模拟的缓存类:提供getCache方法来经过oid查询订单信息; 提供putCache方法将订单信息存放到缓存中;
public class Cache {
private Map<Integer,Order> map = new HashMap();
private volatile static Cache cache;
private Cache(){
}
//双检 + 单例模式
public static Cache getInstance(){
if (cache == null){
synchronized (Cache.class){
if (cache == null){
cache = new Cache();
}
}
}
return cache;
}
//把订单添加到缓存中
public void putCache(int key, Order value){
System.out.println("把订单数据添加到缓存中...");
map.put(key,value);
}
//从缓存中获取订单
public Order getCache(int key){
System.out.println("从缓存中查询订单数据...");
return map.get(key);
}
}
b.定义业务类(被代理的对象类)接口,以供业务类和代理类去实现;spa
public interface OrderDao {
public Order queryOrder(int oid);
}
c.定义业务类、代理类实现业务类(被代理的对象类)接口;
public class OrderDaoImpl implements OrderDao {
@Override
public Order queryOrder(int oid) {
DB db = new DB();
Order order = db.getOrder(oid);
return order;
}
}
d.定义动态代理类工厂,以建立代理对象;
public class ProxyFactory implements InvocationHandler {
private Object target; //被代理的对象
//建立代理对象的方法
public Object create(Object target){
this.target = target;
//JDK 提供动态建立代理类对象的方法(代理对象类加载器,代理对象的接口集合,InvocationHandler(代理对象的处理器))
Object proxy = Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
return proxy;
}
/**
* 生成的代理对象要执行的代理业务方法
* @param proxy:代理类对象
* @param method:被代理对象的方法
* @param args:被代理对象方法的参数
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Order order = null;
if(null != args){
int oid = (int) args[0];
order = Cache.getInstance().getCache(oid);
if(null == order){
//调用真实的业务方法
order = (Order)method.invoke(target,args);
Cache.getInstance().putCache(order.getOid(),order);
}
}
return order;
}
}
测试对比:
public class Test { public static void main(String[] args) { OrderDao od = new OrderDaoImpl(); ProxyFactory proxyFactory = new ProxyFactory(); OrderDao proxy = (OrderDao)proxyFactory.create(od); //建立代理对象,od是被代理对象 Order order = proxy.queryOrder(3); System.out.println(order); System.out.println("------------第二次查询----------"); Order order1 = proxy.queryOrder(3); System.out.println(order1); }}