1. 代理模式主要有兩種:靜態代理和動態代理
2. 靜態代理:
比如要在輸出“HelloWorld”前打印一個字符串“Welcome”
A:先定義一個接口類
- package ttitfly.proxy;
-
- public interface HelloWorld {
- public void print();
-
- }
-
B: 定義一個該接口的實現類
- package ttitfly.proxy;
-
- public class HelloWorldImpl implements HelloWorld{
-
- public void print(){
- System.out.println("HelloWorld");
- }
-
-
-
- }
C:定義一個靜態代理類
- package ttitfly.proxy;
-
- public class StaticProxy implements HelloWorld{
-
- public HelloWorld helloWorld ;
- public StaticProxy(HelloWorld helloWorld){
- this.helloWorld = helloWorld;
- }
-
- public void print(){
- System.out.println("Welcome");
-
- helloWorld.print();
- }
-
-
-
-
-
- }
D: 一個測試類:
- package ttitfly.proxy;
-
- public class TestStaticProxy {
-
- public static void main(String[] args){
- HelloWorld helloWorld = new HelloWorldImpl();
- StaticProxy staticProxy = new StaticProxy(helloWorld);
- staticProxy.print();
-
-
- }
- }
可以看出靜態代理類有一個很不爽的缺點:當如果接口加一個方法(把上面所有的代碼的注釋給去掉),所有的實現類和代理類里都需要做個實現。這就增加了代碼的復雜度。動態代理就可以避免這個缺點。
靜態代理的問題:
1. 一個真實角色必須對應一個代理角色,如果大量使用會導致類的急劇膨脹;
2. 當如果接口加一個方法(比如上面的say),所有的實現類和代理類里都需要做個實現。這就增加了代碼的復雜度;
3.如果事先并不知道真實角色,該如何使用代理呢?
采用動態解決以上問題。
3 。動態代理
動態代理與普通的代理相比較,最大的好處是接口中聲明的所有方法都被轉移到一個集中的方法中處理(invoke),這樣,在接口方法數量比較多的時候,我們可以進行靈活處理,而不需要像靜態代理那樣每一個方法進行中轉。
動態代理類只能代理接口,代理類都需要實現InvocationHandler類,實現invoke方法。該invoke方法就是調用被代理接口的所有方法時需要調用的,該invoke方法返回的值是被代理接口的一個實現類
代理類:
- package ttitfly.proxy;
-
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- import java.lang.reflect.Proxy;
-
- public class DynamicProxy implements InvocationHandler{
-
- private Object object;
-
-
- public Object bindRelation(Object object){
- this.object = object;
- return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(),this);
- }
-
- public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
- System.out.println("Welcome");
- Object result = method.invoke(object, args);
- return result;
- }
-
- }
測試類:
- package ttitfly.proxy;
-
- public class TestDynamicProxy {
- public static void main(String[] args){
- HelloWorld helloWorld = new HelloWorldImpl();
- DynamicProxy dp = new DynamicProxy();
-
- HelloWorld helloWorld1 = (HelloWorld)dp.bindRelation(helloWorld);
- helloWorld1.print();
- helloWorld1.say();
-
-
- HelloWorld helloWorld2 = new HelloWorldImpl();
- helloWorld2.print();
- helloWorld2.say();
-
- }
- }