为对象提供一个替身,以控制对这个对象的访问。即通过代理对象访问目标对象被代理的对象可以是远程对象、创建开销大的对象和需要安全控制的对象代理模式的主要角色
抽象主题:通过接口或抽象类声明真实主题和代理对象实现的业务方法真实主题:实现了抽象主题中的具体业务,是代理对象所代表的真实对象代理:提供了与真实主题相同的接口,其内部含有对真实主题的引用 主要形式有三种
静态代理
静态代理在使用时,需要定义接口或者父类,被代理对象与代理对象一起实现相同的接口或者继承相同的父类静态代理是在代码编译期间就已经存在代理类的字节码文件优点与缺点
优点:在不修改目标对象的功能前提下,能通过代理对象对目标进行扩展缺点:代理对象需要与目标对象实现一样的接口,所以会有很多代理类,一旦接口增加方法,目标对象与代理对象都要维护 代码
package org.demo.proxy;
public class StaticProxyDemo implements SmsService {
private final SmsService smsService;
public StaticProxyDemo(SmsService smsService) {
this.smsService = smsService;
}
@Override
public String send(String message) {
System.out.println("发送短信之前代码");
smsService.send(message);
System.out.println("发送短信之后代码");
return "短信发送成功";
}
}
interface SmsService {
String send(String message);
}
class SmsServiceImpl implements SmsService{
@Override
public String send(String message) {
System.out.println("发送短信:" + message);
return "发送成功";
}
}
class Main {
public static void main(String[] args) {
SmsService smsService = new SmsServiceImpl();
StaticProxyDemo staticProxyDemo = new StaticProxyDemo(smsService);
String re = staticProxyDemo.send("参数");
System.out.println(re);
}
}
动态代理
代理对象不需要实现接口,但是目标对象要实现接口代理对象的生成,是利用JDK的API,动态的在内存中构建代理对象也叫JDK代理和接口代理代码
package org.demo.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class DynamicProxyDemo implements InvocationHandler {
private final Object target;
public DynamicProxyDemo(Object target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("动态代理方法之前操作");
Object invoke = method.invoke(target, args);
System.out.println("动态代理方法之后操作");
return invoke;
}
}
interface DynamicSmsService {
String send(String message);
}
class DynamicSmsServiceImpl implements DynamicSmsService {
@Override
public String send(String message) {
System.out.println("动态代理发送短信");
return null;
}
}
class JdkProxyFactory {
public static Object getProxy(Object target) {
return Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new DynamicProxyDemo(target)
);
}
}
class DynamicMain {
public static void main(String[] args) {
DynamicSmsService smsService = (DynamicSmsService) JdkProxyFactory.getProxy(new DynamicSmsServiceImpl());
smsService.send("java");
}
}
Cglib代理
当目标对象只是一个单独的对象,并没有实现接口时,可以使用目标对象子类来实现代理Cglib是一个高性能的代码生成包,它可以在运行期间扩展java类与实现java接口需要导入Cglib包代码
public class CglibProxyDemo implements MethodInterceptor {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("调用方法之前");
Object invoke = methodProxy.invokeSuper(o, objects);
System.out.println("调用方法之后");
return invoke;
}
}
class CglibSmsService {
public String send(String message) {
System.out.println(message);
return "消息";
}
}
class CglibProxyFactory {
public static Object getProxy(Class> clazz) {
Enhancer enhancer = new Enhancer();
//设置类加载器
enhancer.setClassLoader(clazz.getClassLoader());
//设置被代理类
enhancer.setSuperclass(clazz);
//设置方法拦截器
enhancer.setCallback(new CglibProxyDemo());
return enhancer.create();
}
}
class CglibMain {
public static void main(String[] args) {
CglibSmsService proxy = (CglibSmsService) CglibProxyFactory.getProxy(CglibSmsService.class);
proxy.send("qwe");
}
}



