spring framework入门(6):java中的动态代理(jdk和cglib)

参照:https://www.cnblogs.com/kuotian/p/8849414.html

试验代码:https://download.csdn.net/download/u010476739/11430244

试验环境:

工具:eclipse maven cglib-3.2.12

首先列出java中代理底层的实现方式:

1)静态代理

一个类一个代理,重复代码仍然较多,复用度低。事先写好代理对象类,在程序发布前就已经存在了,调用真实业务方法。

2)jdk动态代理

应用程序发布后,通过动态创建代理对象。通过反射进行invoke回调,要求业务类必须有接口,没有接口无法生成代理对象

3)cglib动态代理

不要求业务类必须有接口,有实现类即可。底层采用asm字节码生成框架生成代理类的字节码,它创建慢,但一旦创建要比jdk方式快。cglib使用继承,因此父类最好不要有final属性,final阻止继承和多态。

一、静态代理

      其实这个就是java中的装饰者模式。

二、jdk动态代理

   2.1 首先定义业务类(注意要被代理的方法抽取到接口中)

Business.java

package jdkproxy;

public interface Business {
	public String Add(int i, int j);
}

BusinessImpl.java

package jdkproxy;

public class BusinessImpl implements Business {
	public String Add(int i, int j) {
		return String.valueOf(i + j);
	}
}

2.2 定义代理类

Logger.java

package jdkproxy;

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

public class Logger implements InvocationHandler{
	private Object targetObject;
    public Logger(Object targetObject) {
        this.targetObject=targetObject;
    }
    
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		Object obj=null;
        try {
           System.out.println("jdk 调用了实例:"+proxy.getClass().toGenericString()+",method:"+method.toGenericString()+".");
            //由目标对象调用实际的功能方法
            obj=method.invoke(targetObject, args);
            System.out.println("jdk 调用完毕!");
        }catch(Exception e) {
            e.printStackTrace();
        }
        return obj;
	}
	
}

2.3 测试调用

public static void main(String[] args) {
		/*
		 * jdk动态代理(基于接口实现): 
		 * 1.业务类只能通过接口被代理,所以被代理的方法要抽取到接口中去定义 
		 * 2.代理类必须实现InvocationHandler接口
		 * 3.使用的时候根据业务类实例对象创建出代理对象,然后用接口调用
		 */
		jdkproxy.Business targetObject = new jdkproxy.BusinessImpl();
		jdkproxy.Business busi = (jdkproxy.Business) Proxy.newProxyInstance(jdkproxy.BusinessImpl.class.getClassLoader(),
				jdkproxy.BusinessImpl.class.getInterfaces(), new jdkproxy.Logger(targetObject));
		System.out.println(busi.Add(1, 2));
}

输出:

三、cglib动态代理

3.1 添加cglib依赖

pom.xml

<project xmlns="http://maven.apache.org/POM/4.0.0"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<groupId>com.jackletter</groupId>
	<artifactId>jdkproxy</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<dependencies>
		<!-- https://mvnrepository.com/artifact/cglib/cglib -->
		<dependency>
			<groupId>cglib</groupId>
			<artifactId>cglib</artifactId>
			<version>3.2.12</version>
		</dependency>
	</dependencies>
</project>

3.2 首先定义业务类(不用接口)

Business.java

package cdlibproxy;

public class Business {
	public String Add(int i, int j) {
		return String.valueOf(i + j);
	}
}

3.3 定义代理类

Logger.java

package cdlibproxy;

import java.lang.reflect.Method;

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

public class Logger implements MethodInterceptor {
	private Object targetObject;
    public Logger(Object targetObject) {
        this.targetObject=targetObject;
    }
	public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
		Object res=null;
        try {
           System.out.println("cglib 调用了实例:"+obj.getClass().toGenericString()+",method:"+method.toGenericString()+".");
            //由目标对象调用实际的功能方法
           res=method.invoke(targetObject, args);
            System.out.println("cglib 调用完毕!");
        }catch(Exception e) {
            e.printStackTrace();
        }
        return res;
	}
}

3.4 测试调用

public static void main(String[] args) {
		/*
		 * cglib动态代理(基于集成实现):
		 * 1.业务类不受任何影响(没有强制抽取接口)
		 * 2.代理类必须实现MethodInterceptor接口
		 * 3.使用的时候根据业务类实例对象创建出代理对象,然后用直接调用
		 */
		cdlibproxy.Business busi2 = new cdlibproxy.Business();
		Object obj = null;
		Enhancer enhancer = new Enhancer();
		enhancer.setSuperclass(busi2.getClass());
		enhancer.setCallback(new cdlibproxy.Logger(busi2));
		busi2 = (cdlibproxy.Business) enhancer.create();
		System.out.println(busi2.Add(2, 3));
}

输出:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

jackletter

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值