1.JAVA的动态代理特征:特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

它是在运行是生成的class对象,在生成时必须提供一组或一个interface给它,然后该class就宣称它实现了这些interface。你当然可以把该class的实例当做这些interface中的任何一个来用,当然,这个DynamicProxy其实就是一个Proxy,他不会替你做实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。因此,DynamicProxy必须实现InvocationHandler接口。
5) 一个动态代理了和一个InvocationHandler 实现关联的。每一个动态代理实例的调用都要通过InvocationHandler接口的handler(调用处理器)来调用,动态代理不做任何执行操作,只是在创建动态代理时,把要实现的接口和handler关联,动态代理要帮助被代理执行的任务,要转交给handler来执行。其实就是调用invoke方法

动态代理:在程序运行时,运用反射机制动态创建而成

2.实用动态代理步骤:

A. 创建一个实现接口InvocationHandler的类,他必须实现invoke方法
B. 创建被代理的类以及接口。
C. 通过Proxy的静态方法newProxyInstance(ClassLoader loader,Class【】interfaces,InvocationHandler handler)创建一个代理
D. 通过代理调用方法

3.jdk动态代理例子:

文件列表:

业务接口:UserService

业务实现:UserServiceImpl

代理类的调用Handler实现:ProxyHandler

JUnit测试类:SpringProxyTest

package com.niewj.service;

import com.niewj.model.User;

public class UserServiceImpl implements UserService {

@Override
 public void add(User user) {
  System.out.println("User Saved. & ");
 }

@Override
 public void delete(User user) {
  System.out.println("User Deleted. &");
 }
}

package com.niewj.service;

import com.niewj.model.User;

public class UserServiceImpl implements UserService {

@Override
 public void add(User user) {
  System.out.println("User Saved. & ");
 }

@Override
 public void delete(User user) {
  System.out.println("User Deleted. &");
 }
}

package com.niewj.service;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

/**
 * JDK动态代理模拟
 *
 * 1.首先明确什么是目标对象target,什么是代理对象proxy!!
 *
 * 2.每个代理对象对象都会有一个相关的InvocationHandler对象。
 * 当代理对象生成的时候,是创建的代理对象,
 * 拿着相关的这个InvocationHandler对象,去自动调Handler类中实现的invoke方法的。
 * 下面一段话来自@javadoc@
 * <p>Each proxy instance has an associated invocation handler.
 * When a method is invoked on a proxy instance, the method
 * invocation is encoded and dispatched to the <code>invoke</code>
 * method of its invocation handler.
 *
 * 3.还有就是我发现,我的Eclipse控制台TMD输出的顺序有误,害的老以为我人品出了什么问题,
 * 不知道控制台的信息是不是不是栈式输出的。(好绕口)
 *
 */
public class ProxyHandler implements InvocationHandler {

// 就是要给这个目标类创建代理对象。
 private Object target;

// 传递代理目标的实例,因为代理处理器需要。也可以用set等方法。
 public ProxyHandler(Object target) {
  this.target = target;
 }

/*
  * 这个方法是给代理对象调用的。
  * 留心的是内部的method调用的对象是目标对象,可别写错。
  */
 @Override
 public Object invoke(Object proxy, Method method, Object[] args)
   throws Throwable {
  Object ret = null;
  // 1.调用前
  cutIntoBefore(method.getName());

ret = method.invoke(target, args);

// 2.调用后
  cutIntoAfter(method.getName());
  return ret;
 }

public void cutIntoBefore(String mName) {
  System.err.println("调用____" + mName + "()____方法之前");
 }

public void cutIntoAfter(String mName) {
  System.err.println("调用____" + mName + "()____方法完后");
 }
}

package com.niewj;

import java.lang.reflect.Proxy;

import org.junit.Test;

import com.niewj.model.User;
import com.niewj.service.ProxyHandler;
import com.niewj.service.UserService;
import com.niewj.service.UserServiceImpl;

public class SpringProxyTest {

@Test
 @SuppressWarnings("rawtypes")
 public void testJDKDynamicProxy() {

/* 1.获取UserServiceImpl对象--目标对象--也就是需要被代理的对象。 */
  UserService userService = new UserServiceImpl();
  // 获取当前类名
  Class clazz = userService.getClass();
  /*
   * 2.获得代理对象。
   * 每一个代理对象都有一个相关的InvocationHandler对象,通过这个handler对象的invoke来实现调用中要完成的自定义猫腻行为
   * 。 可以使用Proxy类和自定义的调用处理逻辑来生成一个代理对象的。
   */
  UserService userServiceProxy = (UserService) Proxy.newProxyInstance(
    clazz.getClassLoader(), clazz.getInterfaces(),
    new ProxyHandler(userService));
  userServiceProxy.add(new User("dotjar"));
  userServiceProxy.delete(new User("DDD"));
 }
}

4.cglib动态代理

JDK实现动态代理需要实现类通过接口定义业务方法,对于没有接口的类,如何实现动态代理呢,这就需要CGLib了。CGLib采用了非常底层的字节码技术,其原理是通过字节码技术为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。JDK动态代理与CGLib动态代理均是实现Spring AOP的基础。

它的运行速度要远远快于JDK的Proxy动态代理。

使用CGLIB需要导入以下两个jar文件:

asm.jar – CGLIB的底层实现。

cglib.jar – CGLIB的核心jar包。

例子:

package com.niewj.service;

import java.lang.reflect.Method;

import com.niewj.model.User;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

public class CgLibProxyImitation implements MethodInterceptor {

private Enhancer enhancer = new Enhancer();

/** 创建代理对象
  * @param targetClass
  * @return 返回代理对象
  */
 @SuppressWarnings("rawtypes")
 public Object getProxy(Class targetClass) {
  enhancer.setSuperclass(targetClass); //设置需要创建子类的类
  enhancer.setCallback(this);
  return enhancer.create(); //通过字节码技术动态创建子类实例

}

public void cutIntoBefore(String mName) {
  System.err.println("调用____" + mName + "()____方法之前");
 }

public void cutIntoAfter(String mName) {
  System.err.println("调用____" + mName + "()____方法完后");
 }

/*  生成的代理对象调用此对象来替代原始对象
  *
  * @java.lang.Object 增强对象
  * @java.lang.reflect.Method 拦截方法
  * @java.lang.Object[] 参数数组
  * @net.sf.cglib.proxy.MethodProxy  用于调用父类,按需要调用多次
  */
 @Override
 public Object intercept(Object obj, Method method, Object[] args,
   MethodProxy proxy) throws Throwable {
  cutIntoBefore(method.getName());
  Object result = proxy.invokeSuper(obj, args);
  cutIntoAfter(method.getName());
  return result;
 }
 
 public static void main(String[] args) {
//  UserService userService = new UserServiceImpl();
  CgLibProxyImitation cglib = new CgLibProxyImitation();
  UserServiceImpl userService = (UserServiceImpl)cglib.getProxy(UserServiceImpl.class);
  userService.add(new User("dotjar"));
  userService.delete(new User("hill"));
 }
}

jdk动态代理与cglib动态代理例子的更多相关文章

  1. 【转载】Spring AOP详解 、 JDK动态代理、CGLib动态代理

    Spring AOP详解 . JDK动态代理.CGLib动态代理  原文地址:https://www.cnblogs.com/kukudelaomao/p/5897893.html AOP是Aspec ...

  2. Java之代理(jdk静态代理,jdk动态代理,cglib动态代理,aop,aspectj)

    一.概念 代理是什么呢?举个例子,一个公司是卖摄像头的,但公司不直接跟用户打交道,而是通过代理商跟用户打交道.如果:公司接口中有一个卖产品的方法,那么公司需要实现这个方法,而代理商也必须实现这个方法. ...

  3. java的静态代理、jdk动态代理和cglib动态代理

    Java的代理就是客户端不再直接和委托类打交道,而是通过一个中间层来访问,这个中间层就是代理.使用代理有两个好处,一是可以隐藏委托类的实现:二是可以实现客户与委托类之间的解耦,在不修改委托类代码的情况 ...

  4. JDK动态代理和CGLib动态代理简单演示

    JDK1.3之后,Java提供了动态代理的技术,允许开发者在运行期间创建接口的代理实例. 一.首先我们进行JDK动态代理的演示. 现在我们有一个简单的业务接口Saying,如下: package te ...

  5. 基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)到底有什么区别。

    基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)到底有什么区别. 我还是喜欢基于Schema风格的Spring事务管理,但也有很多人在用基于@Tras ...

  6. Spring 静态代理+JDK动态代理和CGLIB动态代理

    代理分为两种:静态代理 动态代理 静态代理:本质上会在硬盘上创建一个真正的物理类 动态代理:本质上是在内存中构建出一个类. 如果多个类需要进行方法增强,静态代理则需要创建多个物理类,占用磁盘空间.而动 ...

  7. Spring -- <tx:annotation-driven>注解基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)的区别。

    借鉴:http://jinnianshilongnian.iteye.com/blog/1508018 基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional ...

  8. Spring <tx:annotation-driven>注解 JDK动态代理和CGLIB动态代理 区别。

    基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)到底有什么区别. 我还是喜欢基于Schema风格的Spring事务管理,但也有很多人在用基于@Tras ...

  9. 从静态代理,jdk动态代理到cglib动态代理-一文搞懂代理模式

    从代理模式到动态代理 代理模式是一种理论上非常简单,但是各种地方的实现往往却非常复杂.本文将从代理模式的基本概念出发,探讨代理模式在java领域的应用与实现.读完本文你将get到以下几点: 为什么需要 ...

随机推荐

  1. Linux中使用GoAccess进行日志实时监控

    一.用法命令: goaccess access_log -o /var/www/html/report.html --real-time-html 说明:请先安装Httpd和Goaccess 二.效果 ...

  2. Spring4.2 集成ActiveMQ5.14

    1:libs 2:web.xml <?xml version="1.0" encoding="UTF-8"?> <web-app versio ...

  3. POSTGRESQL 创建表结构、修改字段、导入导出数据库(支持CSV)

    这两个月经常使用postgresql,总结一些经常使用的语句: --创建表 CREATE TABLE customers ( customerid SERIAL primary key , compa ...

  4. 使用Bootstrap+metisMenu完成简单的后台管理界面

    零. 写在前面 作者最近在一个小项目中需要写后台管理界面,在互联网上绕了一圈,最后决定使用Bootstrap+metisMenu来完成.理由1:Bootstrap是目前流行的前端框架,风格简约,简单易 ...

  5. NTCPMSG 开源高性能TCP消息发送组件

    https://www.cnblogs.com/eaglet/archive/2013/01/07/2849010.html 目前的.net 架构下缺乏高效的TCP消息发送组件,而这种组件是构建高性能 ...

  6. 完完全全彻底删除VMware_Workstation

    vmware-workstation,卸载不干净.bat脚本自动处理dll.注册的表垃圾 Download https://pan.baidu.com/s/1dmrOs3rkR8cX5b0vTUOxH ...

  7. 转:如何编译delta3d

    回头看,自上学以来,做虚拟现实.三维地理信息系统已有六个年头.为了节省花费在编码上的精力,编程应用的API函数也由opengl变成了OpenSceneGraph,但还是花费了很多气力,为了更好的提高开 ...

  8. android应用Theme(二)

    另外一种实现android应用Theme的方式是通过apk来实现的. 以下是一个demo. 1.首先必须新建一个apk.类似的插件,然后在该apk的AndroidManifest.xml文件的appl ...

  9. VS2017写的exe调用Delphi 7写的DLL

    公司有个很古老的系统,代码量很大,并且稳定线上运行10几年,这系统是公司的核心,公司收入基本靠它,系统几乎都是Delphi 7写的,要重写是不可能的.因为Delphi 7编译出来的DLL默认的导出符号 ...

  10. windows下Oracle Tuxedo编译应用前需要配置的相关环境变量

    rem (c) BEA Systems, Inc. All Rights Reserved. rem Copyright (c) BEA Systems, Inc. rem All Rights Re ...