1. 程式人生 > >Java 代理模式

Java 代理模式

all object 控制 cglib動態代理 per 使用 sets back 概念

代理模式的概念:為其他對象提供一種代理一控制對這個對象的訪問。

應用場景:在開發中一些業務類中方法,除了需求的操作,還需要進行其他的非業務操作。比如提供給app的接口。除了接口要實現的業務邏輯,還要對用戶的信息,設備的信息進行驗證,參數的加密解密。這種在每個接口方法前都要調用的,和業務代碼參在一起就會重復,累贅。可以通過代理類將這些經常使用的方法抽取出來。業務類只用關心業務操作,降低耦合。

靜態代理

代理和被代理對象在代理前是確定的。他們都實現了相同的接口或者繼承了相同的抽象類。

動態代理

動態產生代理,實現對不同類,不同方法進行代理

1) JDK動態代理

java提供了動態代理技術,允許開發者在運行期創建接口的代理實例。JDK動態代理主要涉及到java.lang.reflect包下的Proxy和InvocationHandler.

接口

public interface TrafficTools{
    
    public void drive();

}

實現

public class Car implements TrafficTools {

    @Override
    public void drive() {
        System.out.println("汽車 行駛中。。。");
    }

}

代理類

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class CarProxy implements InvocationHandler{
    
    public CarProxy(Object target) {
        super();
        this.target = target;
    }

    private Object target;

    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
            
throws Throwable { long start = System.currentTimeMillis(); System.out.println("汽車開始行駛。。。"); method.invoke(target); long end = System.currentTimeMillis(); System.out.println("汽車結束行駛。。。"+(end - start)); return null; } }

測試代理

import java.lang.reflect.Proxy;

public class Test {
    
    public static void main(String[] args) {
        Car car = new Car();
        CarProxy h = new CarProxy(car);
        Class<?> cls = car.getClass();
        TrafficTools t = (TrafficTools)Proxy.newProxyInstance(cls.getClassLoader(), 
                 cls.getInterfaces(), h);
        t.drive();
    }

}

2)CGLib動態代理

JDK動態代理只能為接口創建動態代理,CGLib采用非常底層的字節碼技術,可以為一個類創建代理。需要引用cglib的jar包。

import java.lang.reflect.Method;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

public class CglibProxy implements MethodInterceptor{
    
    private Enhancer enhancer = new Enhancer();
    
    public Object getProxy(Class clazz){
        enhancer.setSuperclass(clazz);
        enhancer.setCallback(this);
        return enhancer.create();
    }

    /***
     * 攔截所有目標類的方法調用
     */
    @Override
    public Object intercept(Object obj, Method m, Object[] args,
            MethodProxy proxy) throws Throwable {
        return proxy.invokeSuper(obj, args);
    }

}

CGLib所創建的動態代理對象的性能比JDK的所創建的代理對象的性能高很多。但CGLIb在創建代理對象時所花費的時間比JDK動態代理多。

Java 代理模式