1.代理模式的原理:

使用一个代理将对象包装起来, 然后用该代理对象取代原始对象。任何对原始对象的调用都要通过代理。代理对象决定是否以及何时将方法调用转到原始对象上。

2.静态代理

2.1 举例:

实现Runnable接口的方法创建多线程。
Class MyThread implements Runnable{} //相当于被代理类
Class Thread implements Runnable{} //相当于代理类
main(){
MyThread t = new MyThread();
Thread thread = new Thread(t);
thread.start();//启动线程;调用线程的run()
}

2.2 静态代理的缺点:

① 代理类和目标对象的类都是在编译期间确定下来,不利于程序的扩展。
② 每一个代理类只能为一个接口服务,这样一来程序开发中必然产生过多的代理。

3.动态代理的特点:

动态代理是指客户通过代理类来调用其它对象的方法,并且是在程序运行时根据需要动态创建目标类的代理对
象。

4.动态代理的实现

4.1 需要解决的两个主要问题:

  • 问题一:如何根据加载到内存中的被代理类,动态的创建一个代理类及其对象。

(通过Proxy.newProxyInstance()实现)

  • 问题二:当通过代理类的对象调用方法a时,如何动态的去调用被代理类中的同名方法a。

(通过InvocationHandler接口的实现类及其方法invoke())

4.2 代码实现:

  1. /**
  2. *
  3. * 动态代理的举例
  4. *
  5. * @author shkstart
  6. * @create 2019 上午 10:18
  7. */
  8. interface Human{
  9. String getBelief();
  10. void eat(String food);
  11. }
  12. //被代理类
  13. class SuperMan implements Human{
  14. @Override
  15. public String getBelief() {
  16. return "I believe I can fly!";
  17. }
  18. @Override
  19. public void eat(String food) {
  20. System.out.println("我喜欢吃" + food);
  21. }
  22. }
  23. class HumanUtil{
  24. public void method1(){
  25. System.out.println("====================通用方法一====================");
  26. }
  27. public void method2(){
  28. System.out.println("====================通用方法二====================");
  29. }
  30. }
  31. class ProxyFactory{
  32. //调用此方法,返回一个代理类的对象。解决问题一
  33. public static Object getProxyInstance(Object obj){//obj:被代理类的对象
  34. MyInvocationHandler handler = new MyInvocationHandler();
  35. handler.bind(obj);
  36. return Proxy.newProxyInstance(obj.getClass().getClassLoader(),obj.getClass().getInterfaces(),handler);
  37. }
  38. }
  39. class MyInvocationHandler implements InvocationHandler{
  40. private Object obj;//需要使用被代理类的对象进行赋值
  41. public void bind(Object obj){
  42. this.obj = obj;
  43. }
  44. //当我们通过代理类的对象,调用方法a时,就会自动的调用如下的方法:invoke()
  45. //将被代理类要执行的方法a的功能就声明在invoke()中
  46. @Override
  47. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  48. HumanUtil util = new HumanUtil();
  49. util.method1();
  50. //method:即为代理类对象调用的方法,此方法也就作为了被代理类对象要调用的方法
  51. //obj:被代理类的对象
  52. Object returnValue = method.invoke(obj,args);
  53. util.method2();
  54. //上述方法的返回值就作为当前类中的invoke()的返回值。
  55. return returnValue;
  56. }
  57. }
  58. public class ProxyTest {
  59. public static void main(String[] args) {
  60. SuperMan superMan = new SuperMan();
  61. //proxyInstance:代理类的对象
  62. Human proxyInstance = (Human) ProxyFactory.getProxyInstance(superMan);
  63. //当通过代理类对象调用方法时,会自动的调用被代理类中同名的方法
  64. String belief = proxyInstance.getBelief();
  65. System.out.println(belief);
  66. proxyInstance.eat("四川麻辣烫");
  67. System.out.println("*****************************");
  68. NikeClothFactory nikeClothFactory = new NikeClothFactory();
  69. ClothFactory proxyClothFactory = (ClothFactory) ProxyFactory.getProxyInstance(nikeClothFactory);
  70. proxyClothFactory.produceCloth();
  71. }
  72. }
  73. 体会:反射的动态性。