什么是动态代理?
无侵入式地给代码增加额外功能
为什么需要代理?
对象如果嫌身上干的事太多的话,可以通过代理来转移部分职责
动态代理的实现步骤:
一、定义接口
//代码中定义的是需要代理的方法
public interface Star {
public abstract String sing(String name);
public abstract void dance();
}
二、定义接口实现类
public class BigStart implements Star{
private String name;
public BigStart(){}
public BigStart(String name) {
this.name = name;
}
@Override
public String sing(String name){
System.out.println(this.name + "正在唱" + name);
return "谢谢";
}
@Override
public void dance(){
System.out.println(this.name + "正在跳舞");
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
三、新建类用于创建代理
/**
* 类的作用:
* 创建一个代理
*/
public class ProxyUtil {
/**
* 方法的作用:
* 给一个明星的对象,创建一个代理
* 形参:
* 被代理的明星对象
* 返回值:
* 给明星创建的代理
* 需求:
* 外面的人想要大明星唱一首歌
* 1.获取代理的对象
* 代理对象 = ProxyUtil.createProxy(大明星的对象);
* 2.再调用代理的唱歌方法
* 代理对象.唱歌的方法();
* @param bigStart
*/
public static Star createProxy(BigStart bigStart){
/**
* java.lang.reflect.Proxy类:提供了为对象产生代理对象的方法
*/
Star star = (Star) Proxy.newProxyInstance(
ProxyUtil.class.getClassLoader(), //参数一:用于指定用哪个类加载器,去加载生成的代理类
new Class[]{Star.class}, //参数二:指定接口,这些接口用于指定生成的代理长什么样,也就是该有哪些方法
new InvocationHandler() { //参数三:用来指定生成的代理对象要干什么事情
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
/**
* 参数一:代理的对象
* 参数二:要运行的方法
* 参数三:调用方法时,传递的实参
*/
if("sing".equals(method.getName())){
System.out.println("准备话筒,收钱");
}else if("dance".equals(method.getName())){
System.out.println("准备场地,收钱");
}
//去找大明星开始唱歌跳舞
return method.invoke(bigStart,args);
}
}
);
return star;
}
}
四、测试
public class Test {
public static void main(String[] args) {
/**
* 需求:
* 外面的人想要大明星唱一首歌
* 1.获取代理的对象
* 代理对象 = ProxyUtil.createProxy(大明星的对象);
* 2.再调用代理的唱歌方法
* 代理对象.唱歌的方法();
*/
//1.获取代理对象
BigStart bigStart = new BigStart("yeye");
Star proxy = ProxyUtil.createProxy(bigStart);
//2.调用唱歌方法
System.out.println(proxy.sing("思念是一种病"));
proxy.dance();
}
}