黑马程序员:Java基础总结



静态代理模式&动态代理

 
ASP.Net+Android+IO开发

.Net培训
、期待与您交流!




静态代理模式
public 
class 
Ts {
       
public 
static 
void 
main(String[] args) 
throws 
Exception {
             
// 通过中介公司生产一批衣服
            ClothingProduct cp = 
new 
ProxCompany(
new 
LiNingCompany());
            cp.productClothing();
      }

}

/**
 * 定义生产一批衣服功能的接口
 *
 */
interface 
ClothingProduct {

       
void 
productClothing(); 
// 有生产一批衣服的功能

}

/**
 *
 * 代理类:中介公司
 *
 */
class 
ProxCompany 
implements 
ClothingProduct {

       
private 
ClothingProduct 
cp 

// 中介公司不会生产衣服,需要找一家真正能生产衣服的公司

      ProxCompany(ClothingProduct cp) {
             
super
();
             
this
.
cp 
= cp;
      }

       
@Override
       
public 
void 
productClothing() {

            System. 
out
.println(
"收取1000块钱的中介费" 
);
             
cp
.productClothing();

      }

}

/**
 *
 * 李宁公司是生产服装的目标类
 *
 */
class 
LiNingCompany 
implements 
ClothingProduct {

       
@Override
       
public 
void 
productClothing() {

            System. 
out
.println(
"生产一批衣服。。。。" 
);
      }

}


上面程序的做法,使用的模式是静态代理模式

静态代理模式在现实编程中的弊端:

它的特征是代理类和目标对象的类都是在编译期间确定下来的,不利于程序上的扩展,上面示例中,如果客户还想找一个“生产一批鞋子”的工厂,那么还需要新增加一个代理类和一个目标类。如果客户还需要很多其他的服务,就必须一一的添加代理类和目标类。那就需要写很多的代理类和目标类


动态代理技术


java.lang.reflect
类 Proxy

Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。

构造方法摘要
protected Proxy(InvocationHandler h)
          使用其调用处理程序的指定值从子类(通常为动态代理类)构建新的 Proxy 实例。
字段摘要
protected  InvocationHandler h
          此代理实例的调用处理程序。
方法摘要
static InvocationHandler getInvocationHandler(Object proxy)
          返回指定代理实例的调用处理程序。
static Class<?> getProxyClass(ClassLoader loader, Class<?>... interfaces)
          返回代理类的 java.lang.Class 对象,并向其提供类加载器和接口数组。
static boolean isProxyClass(Class<?> cl)
          当且仅当指定的类通过 getProxyClass 方法或 newProxyInstance 方法动态生成为代理类时,返回 true。
static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
          返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。

每个代理实例都具有一个关联的调用处理程序。对代理实例调用方法时,将对方法调用进行编码并将其指派到它的调用处理程序的 invoke 方法。

public interface InvocationHandler 方法摘要
 Object invoke(Object proxy, Method method, Object[] args)
          在代理实例上处理方法调用并返回结果。
参数:
proxy - 在其上调用方法的代理实例
method - 对应于在代理实例上调用的接口方法的 
Method 实例。
Method 对象的声明类将是在其中声明方法的接口,该接口可以是代理类赖以继承方法的代理接口的超接口。
args - 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 
null。基本类型的参数被包装在适当基本包装器类(如 
java.lang.Integer 或 
java.lang.Boolean)的实例中。
返回:
从代理实例的方法调用返回的值。如果接口方法的声明返回类型是基本类型,则此方法返回的值一定是相应基本包装对象类的实例;否则,它一定是可分配到声明返回类型的类型。如果此方法返回的值为 
null 并且接口方法的返回类型是基本类型,则代理实例上的方法调用将抛出 
NullPointerException。否则,如果此方法返回的值与上述接口方法的声明返回类型不兼容,则代理实例上的方法调用将抛出 
ClassCastException

import 
java.lang.reflect.InvocationHandler;
import 
java.lang.reflect.Method;
import 
java.lang.reflect.Proxy;
import 
java.util.ArrayList;
import 
java.util.Collection;

public 
class 
Ts {
       
public 
static 
void 
main(String[] args) 
throws 
Exception {
             ArrayList test = 
new 
ArrayList();
             
// 代理只可以强转换成接口
             Collection list = (Collection ) getProxy(test, 
new 
MyAdvice());
             list.add(123);
             list.add(123);
            System. 
out
.println(list);

      }

       
public 
static 
Object getProxy(
final 
Object target, 
final 
Advice adv)
/* 终态是因为内部类调用 */ 
{
             
// 返回一个指定接口的代理类实例 obj
            Object obj = Proxy. newProxyInstance(
             
// 定义代理类的类加载器
                        target.getClass().getClassLoader(),
                         
// 代理类要实现的接口列表
                        target.getClass().getInterfaces(),
                         
// 指派方法调用的调用处理程序 InvocationHandler
                         
new 
InvocationHandler() {

                               
@Override
                               
public 
Object invoke(Object proxy, Method method,
                                          Object[] args) 
throws 
Throwable {                         
                                    adv.begintime(method);
                                     
//target目标 ,args方法参数,调用原来的方法
                                    Object retVal = method.invoke(target, args);
                                    adv.endtime(method);
                                     
return 
retVal;
                              }
                        });

             
return 
obj;
      }
}
//插入的建议接口
interface 
Advice {
       
void 
begintime(Method method);

       
void 
endtime(Method method);
}
//我的建议
class 
MyAdvice 
implements 
Advice {

       
@Override
       
public 
void 
begintime(Method method) {
            Long time = System. currentTimeMillis();
            System. 
out
.println(method.getName() + time);
      }

       
@Override
       
public 
void 
endtime(Method method) {
            Long time = System. currentTimeMillis();
            System. 
out
.println(method.getName() + time);
      }

}

一个更巧妙的方法:自定义一个处理程序
public 
class 
Ts {
       
public 
static 
void 
main(String[] args) 
throws 
Exception {
            ProxyHandler handler= 
new 
ProxyHandler();
             ClothingProduct cp2=(ClothingProduct)handler.newProxyInstance( 
new 
LiNingCompany());
            cp2.productClothing();

      }

}

class 
ProxyHandler 
implements 
InvocationHandler {

       
/* 目标对象 */
       
private 
Object 
target 
;

       
/* 创建目标对象的代理对象 */
       
public 
Object newProxyInstance(Object target) {

             
this
.
target 
= target;

             
/*
             * 第一个参数:定义代理类的类加载器
             * 第二个参数:代理类要实现的接口 列表
             * 第三个参数:指派方法调用的调用处理程序
             */
             
return 
Proxy.newProxyInstance(
this

target
.getClass().getClassLoader(),
                         
this
.
target 
.getClass().getClasses(), 
this
);

      }

       
@Override
       
public 
Object invoke(Object proxy, Method method, Object[] args)
                   
throws 
Throwable {

            Object result = 
null
;

            System. 
out
.println(
"目标对象上的方法调用之前可以添加其他代码。。。" 
);
            result = method.invoke( 
this
.
target 
, args); 
// 通过反射调用目标对象上的方法
            System. 
out
.println(
"目标对象上的方法调用之后可以添加其他代码。。。" 
);

             
return 
result;
      }

}





 
ASP.Net+Android+IO开发

.Net培训
、期待与您交流!

黑马程序员:Java基础总结----静态代理模式&动态代理的更多相关文章

  1. 黑马程序员----java基础笔记中(毕向东)

    <p>------<a href="http://www.itheima.com" target="blank">Java培训.Andr ...

  2. 黑马程序员——JAVA基础之简述设计模式

    ------- android培训.java培训.期待与您交流! ---------- 设计模式(Design Patterns) 设计模式(Design pattern)是一套被反复使用.多数人知晓 ...

  3. 黑马程序员Java基础班+就业班课程笔记全发布(持续更新)

    正在黑马学习,整理了一些课程知识点和比较重要的内容分享给大家,也是给自己拓宽一些视野,仅供大家交流学习,大家有什么更好的内容可以发给我 ,现有黑马教程2000G  QQ 1481135711 这是我总 ...

  4. 黑马程序员----java基础笔记上(毕向东)

    ------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 笔记一共记录了毕向东的java基础的25天课程,分上.中.下 本片为上篇,涵盖前10天课程 1. ...

  5. 黑马程序员_java基础笔记(13)...类加载器和代理

    —————————— ASP.Net+Android+IOS开发..Net培训.期待与您交流! —————————— 1,类加载器.2,代理. 1,类加载器. Java虚拟机中可以安装多个类加载器,系 ...

  6. 黑马程序员——JAVA基础之泛型和通配符

    ------- android培训.java培训.期待与您交流! ---------- 泛型:            JDK1.5版本以后出现新特性.用于解决安全问题,是一个类型安全机制. 泛型好处: ...

  7. 黑马程序员——JAVA基础之简述面向对象,类,变量,匿名对象

    ------- android培训.java培训.期待与您交流! ---------- 面向对象: 面向对象是相对面向过程而言 面向对象和面向过程都是一种思想 面向过程 强调的是功能行为 面向对象 将 ...

  8. 黑马程序员——JAVA基础之语法、命名规则

    ------- android培训.java培训.期待与您交流! ---------- 1.java语言组成:关键字,标识符,注释,常量和变量,运算符,语句,函数,数组. 2.java关键字:被Jav ...

  9. 黑马程序员——JAVA基础之主函数main和静态static,静态代码块

    ------- android培训.java培训.期待与您交流! ---------- 主函数:是一个特殊的函数.作为程序的入口,可以被jvm调用. 主函数的定义: public:代表着该函数访问权限 ...

  10. 黑马程序员——JAVA基础之多线程的安全问题

    ------- android培训.java培训.期待与您交流! ---------- 导致多线程出现问题的一个特殊的状态:就绪.具备了执行资格,但是还没有获取资源. 导致安全问题的出现的原因: 1. ...

随机推荐

  1. 多文件上传组件FineUploader使用心得

    原文 多文件上传组件FineUploader使用心得 做Web开发的童鞋都知道,需要经常从客户端上传文件到服务端,当然,你可以使用<input type="file"/> ...

  2. VC 无标题栏对话框移动

    操作系统:Windows 7软件环境:Visual C++ 2008 SP1本次目的:实现无框移动 所谓的无标题栏对话框,是基于对话框的工程,对话框属性Border设置为None,对话框如下所示: 为 ...

  3. 【Unity技巧】使用单例模式Singleton

    这几天想把在实习里碰到的一些好的技巧写在这里,也算是对实习的一个总结.好啦,今天要讲的是在Unity里应用一种非常有名的设计模式——单例模式. 开场白 单例模式的简单介绍请看前面的链接,当然网上还有很 ...

  4. java layout 表格项增加、删除、修改

    实现的内容为:点击表格某项,再点击删除钮,可实现删除点击表格某项,再点击编辑按钮,可实现内容改变点击添加按钮,可实现向表格中添加内容 总结:总的来说中间遇到了很多困难,但是都一步步的解决了. pack ...

  5. Android中Parcelable序列化总结

    在使用Parcelable对android中数据的序列化操作还是比较有用的,有人做过通过对比Serializable和Parcelable在android中序列化操作对象的速度比对,大概Parcela ...

  6. 基于visual Studio2013解决C语言竞赛题之1054抽牌游戏

       题目 解决代码及点评 /************************************************************************/ /* 54 ...

  7. [置顶] js中如何复制一个对象,如何获取所有属性和属性对应的值

    在js中如何复制一个对象,例如如下一个js对象. 如果知道这个对象的所有属性自然就可以重新new一个,然后对每个属性赋值,就可以做到,但如果不知道呢?如何创建一个内容相同 的对象呢? var obj= ...

  8. Kendo UI开发教程(22): Kendo MVVM 数据绑定(十一) Value

    Value绑定可以把ViewModel的某个属性绑定到DOM元素或某个UI组件的Value属性.当用户修改DOM元素或UI组件的值时,绑定的ViewModel的值也随之发生改名.同样,如果ViewMo ...

  9. 关于CopyU!的常见问题解答

    拷优(CopyU!)常见问题解答 本常见问题解答列举了一些常见的疑问及其解释,如果您对CopyU!有任何问题,请您首先查看本解答! 本解答将会保持随时更新! 一.使用篇:     1.问:我的杀毒软件 ...

  10. 解决删除Volume报错的问题

    很久没有遇到过删除Volume出错使得Volume处于Error_Deleting状态的情况了,昨天删除一个Volume时又出现了这个问题,这里顺便把解决方法记录一下. 注意我这里针对的是后端采用is ...