1. OvervIEw
Java在Java.lang.reflect包下,定義了自己的代理。利用這個包下的類,我們可以在運行時動態地創建一個代理類,實現一個或多個接口。並將方法的調用轉發到你所指定的類。因為實際代理是在運行時創建的,所以稱為:動態代理。
Proxy:完全由Java產生的,而且實現了完整的subject接口。
InvocationHandler:Proxy上的任何方法調用都會被傳入此類,InvocationHandler控制對RealSubject的訪問。
因為Java已經幫助我們創建了Proxy類,我們需要有辦法告訴Proxy類你要做什麼,我們不能像以前一樣把代碼寫入到Proxy類中,因為Proxy類不是我們實現的。那麼我們應該放在哪裡?放在InvocationHandler類中,InvocationHandler類是響應代理的任何調用。我們可以吧InvocationHandler想成是代理收到方法調用後,請求做實際工作的對象。
2. Java.lang.reflect.InvocationHandler
被代理實例所實現的一個接口,內部只有一個invoke()方法,簽名如下;
Java代碼
- public Object invoke(Object proxy, Method method, Object[] args)
當代理的方法被調用的時候,代理就會把這個調用轉發給InvocationHandler,也就會調用它的invoke()方法。
3. Java.lang.reflect.Proxy
提供用於創建動態代理類和實例的靜態方法,它還是由這些方法創建的所有動態代理類的超類,我們經常使用的靜態方式是:
Java代碼
- newProxyInstance(ClassLoader loader, Class>[] interfaces, InvocationHandler h)
4. 示例:
情形:自己可以查看修改姓名性別,但是不能修改rate。他人可以查看姓名,性別以及修改rate,但是不能修改姓名性別。
4.1 定義一個接口:
Java代碼
- public interface Person {
- String getName();
- String getGender();
- void setName(String name);
- void setGender(String gender);
- void setRate(int rate);
- int getRate();
- }
4.2 定義實現Person接口類
Java代碼
- public class PersonImpl implements Person {
- String name;
- String gender;
- String interests;
- int rate;
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getGender() {
- return gender;
- }
- public void setGender(String gender) {
- this.gender = gender;
- }
- public String getInterests() {
- return interests;
- }
- public void setInterests(String interests) {
- this.interests = interests;
- }
- public int getRate() {
- return rate;
- }
- public void setRate(int rate) {
- this.rate = rate;
- }
4.3 定義OwnerInvocationHandler類,表示如果為本人,則可以進行修改查看姓名性別。
Java代碼
- public class OwnerInvocationHandler implements InvocationHandler{
- private Person personBean;
- public OwnerInvocationHandler(Person personBean){
- this.personBean = personBean;
- }
- @Override
- public Object invoke(Object proxy, Method method, Object[] args)
- throws IllegalAccessException {
- try {
- if(method.getName().startsWith("get")){//如果方法名為get,就調用person類內的get相應方法
- return method.invoke(personBean, args);
- }else if(method.getName().equals("setRate")){ // 如果方法是setRate,則拋出異常
- throw new IllegalAccessException("Access deny");
- }else if(method.getName().startsWith("set")){ //如果為set,就調用person類內的set相應方法
- return method.invoke(personBean, args);
- }else {
- System.out.println("non method invoke");
- }
- } catch (InvocationTargetException e) {
- e.printStackTrace();
- }
- return null;
- }
- }
4.4 定義NonInvocationHandler類,表示如果不為本人,則可以進行查看姓名性別和修改rate。
Java代碼
- public class NonInvocationHandler implements InvocationHandler{
- //
- private Person person;
- public NonInvocationHandler(Person person){
- this.person = person;
- }
- @Override
- public Object invoke(Object proxy, Method method, Object[] args)
- throws Throwable {
- if(method.getName().startsWith("setRate")){
- return method.invoke(person, args);
- }else if (method.getName().startsWith("get")){
- return method.invoke(person, args);
- } else {
- System.out.println("non method invoke");
- return null;
- }
- }
- }
4.5 測試類MyDynamicProxy
Java代碼
- public class MyDynamicProxy {
- public Person getOwnerPersonBeanProxy(Person person){
- return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(),
- person.getClass().getInterfaces(), new OwnerInvocationHandler(person));
- }
- public Person getNonPersonBeanProxy(Person person){
- return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(),
- person.getClass().getInterfaces(), new NonInvocationHandler(person));
- }
- public static void main(String[] args) {
- MyDynamicProxy mdp = new MyDynamicProxy();
- mdp.test();
- }
- public void test(){
- //
- Person person = getPersonBeanFromDB1();
- Person personProxy = getOwnerPersonBeanProxy(person);
- System.out.println(personProxy.getName());
- try {
- personProxy.setRate(2);
- } catch (Exception e) {
- System.out.println("can not setRate");
- }
- //
- Person person1 = getPersonBeanFromDB1();
- Person personProxy2 = getNonPersonBeanProxy(person1);
- System.out.println(personProxy2.getName());
- personProxy2.setRate(2);
- System.out.println(personProxy2.getRate());
- }
- private Person getPersonBeanFromDB1(){
- Person pb = new PersonImpl();
- pb.setName("remy");
- pb.setGender("girl");
- pb.setRate(1);
- return pb;
- }
輸出結果:
Java代碼
- remy
- can not setRate
- remy
- 2