❶ JDK动态代理
JDK动态代理是基于接口使用,通过Proxy.newProxyInstance方法使用。
传入参数:当前类加载器、目标接口、带目标类的InvocationHandle实现类
1、Proxy#newProxyInstance
2、Constructor#newInstance
4、sun.reflect.NativeConstructorAccessorImpl#newInstance
4、sun.reflect.NativeConstructorAccessorImpl#newInstance
ReflectionFactory.inflationThreshold 默认值是15
5、native的newInstance0
3、sun.reflect.#newInstance
4、执行GeneratedConstructorAccessor3代理类方法
第16次调用generate方法生成代理类GeneratesConstructorAccessor,第17次执行GeneratesConstructorAccessor代理类方法
反射类加载器会导致Perm溢出
ConstructorAccessor 有下面实现类
1、NativeConstructorAccessorImpl
JNI(java本地接口)
2、
注入ConstructorAccessorImpl属性(值是1或3),实现newInstance
3、GeneratedConstructorAccessorX
通过代理类
MethodAccessor 有下面实现类
1、NativeMethodAccessorImpl
NativeMethodAccessorImpl是通过JNI(java本地接口)存取器获取反射类字节码信息
2、DelegatingMethodAccessorImpl
DelegatingMethodAccessorImpl注入MethodAccessorImpl属性(值是1或3),实现invoke调用
3、GeneratedMethodAccessorX
通过代理类加载反射类字节码
NativeConstructorAccessorImpl或NativeMethodAccessorImpl都有一个字段numInvocations。用来计算调用JNI次数
当大于15就会生成GeneratedConstructorAccessorX或GeneratedConstructorAccessorX的字节码类文件,该字节码通过DelegatingClassLoader类加载。
为什么设计这种机制:为了提高效率
MethodAccessorGenerator类
generate方法实现了
1、asm生成字节码数组
2、defineClass生成DelegatingClassLoader代理类加载器
3、generateName方法生成字节码文件GeneratedXXXAccessorX
Inflation机制提高了反射的性能,但是对于重度使用有两个问题
1、初次加载性能较低
2、动态加载的字节码导致PermGen持续增长
JDK是基于接口实现
Inflation机制
关键是生成代理类和调用inovke方法的实现方式
1、JNI
2、asm字节码
❷ Spring的两种代理JDK和CGLIB的区别
原理区别:x0dx0ajava动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。x0dx0a1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP x0dx0a2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP x0dx0a3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换
❸ 静态代理,JDK动态代理和CGLib动态代理之前的区别
1、静态代理:静态代理中的代理类,需要我们自己写
JDK动态代理类实现了InvocationHandler接口。在重写的invoke方法中可以看出,JDK动态代理的基础是反射(method.invoke(对象,参数)),还好反射看的比较多,到现在还记得。在这里需要提到的是Proxy.newProxyInstance(),这个方法。字面上的意思是 新建一个代理类的实例,这一点就和静态代理不同了。里面的参数有三个 类加载器、所有的接口,得到InvocationHandler接口的子类实例。这就是JDK动态代理,该代理有以下几种特点:
1、Interface:对于JDK Proxy,业务类是需要一个Interface的,这是一个缺陷;
2、Proxy:Proxy类是动态产生的,这个类在调用Proxy.newProxyInstance()方法之后,产生一个Proxy类的实力。实际上,这个Proxy类也是存在的,不仅仅是类的实例,这个Proxy类可以保存在硬盘上;
3、Method:对于业务委托类的每个方法,现在Proxy类里面都不用静态显示出来
4、InvocationHandler:这个类在业务委托类执行时,会先调用invoke方法。invoke方法在执行想要的代理操作,可以实现对业务方法的再包装。
以上就是JDK动态代理
3、CGLib动态代理:上面的JDK Proxy只能代理实现了接口的类,而不能实现接口的类就不能实现JDK代理。这时候就需要CGLib动态代理类
这里需要注意的是实现MethodIntercetor接口,必须导入cglib-nodep-2.1_3.jar这个包。CGLib是针对类来实现代理的,他的原理是对指定的目标生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
❹ Spring中默认使用jdk代理还是cglib代理
jdk代理(基于接口):如果目标实现了接口,那么默认使用jdk代理。
cglib代理(基于类):如果目标没有实现接口,那么默认使用cglib代理。
spring中的配置proxy-target-class 属性:
proxy-target-class:true 设置基于类的代理,cglib代理。
proxy-target-class:false 设置基于接口的代理,jdk代理,但如果目标没有实现接口,使用cglib代理。
如果当前类方法上有 @Cacheable @Transactional 注解或者 被aop加强的话 生成的是代理对象,具体哪种代理看上面的区分,如果没有上述条件的话那么创建的对象是 原始对象。
❺ Java代理的作用和实现
JDK 动态代理
动态代理的核心其实就是代理对象的生成,即 Proxy.newProxyInstance(classLoader, proxyInterface, handler)。
让我们进入newProxyInstance方法观摩下,核心代码其实就三行。
这个方法需要三个参数:
ClassLoader,用于加载代理类的 Loader 类,通常这个 Loader 和被代理的类是同一个 Loader 类。
Interfaces,是要被代理的那些那些接口。
InvocationHandler,就是用于执行除了被代理接口中方法之外的用户自定义的操作,也是用户需要代理的最终目的。用户调用目标方法都被代理到 InvocationHandler 类中定义的唯一方法 invoke 中。
一个典型的动态代理创建对象过程可分为以下四个步骤:
1、通过实现InvocationHandler接口创建调用处理器
2、通过为Proxy类指定ClassLoader对象和一组interface创建动态代理类
3、通过反射机制获取动态代理类的构造函数,其参数类型是调用处理器接口类型
4、通过构造函数创建代理类实例,此时需将调用处理器对象作为参数被传入
为了简化对象创建过程,Proxy类中的newProxyInstance方法封装了2~4,只需两步即可完成代理对象的创建。
生成的proxySubject继承Proxy类实现Subject接口。实现的Subject的方法实际是调用处理器的invoke方法,而invoke方法利用反射调用的是被代理对象的方法(Object result=method.invoke(proxied,args));
重点Proxy.newProxyInstance,源码分析,会在其他文档中单独总结记录。类Proxy的getProxyClass方法调用ProxyGenerator的 generateProxyClass方法产生ProxySubject.class的二进制数据。
创建代理对象时序图
获取代理类
getProxyClass(loader, interfaces)方法用于获取代理类,它主要做了三件事情:
在当前类加载器的缓存里搜索是否有代理类,没有则生成代理类并缓存在本地JVM里。
生成并加载代理类
代理类的生成主要是以下这两行代码:
代理类的生成过程
ProxyGenerator.generateProxyClass()方法属于sun.misc包下,Oracle并没有提供源代码,但是我们可以使用
JD-GUI这样的反编译软件打开jrelib
t.jar来一探究竟,以下是其核心代码的分析。
生成的代理类源码
那么通过以上分析,我们可以推出动态代理为我们生成了一个这样的代理类。把方法doSomeThing的方法体修改为调用LogInvocationHandler的invoke方法。
测试代理的代码如下:
下面看一个自定义代理的实现。
被代理类接口
被代理类
调用处理器(切面)
测试我们的代理实现
运行结果:
Proxy 接口
Proxy 的主要静态变量
ProxySubject 源码
创建的代理类 ProxySubject.class
CGLib 动态代理
动态字节码生成。使用动态字节码生成技术实现AOP原理是在运行期间目标字节码加载后,生成目标类的子类,将切面逻辑加入到子类中,所以使用Cglib实现AOP不需要基于接口。
❻ JDK动态代理简单实现
1.jdk动态代理目标类必须实现一个或多个接口 如果没有实现接口则无法实现jdk动态代理,如果想实现没有接口的类,就可以使用cglib 代理(子类代理)
2.实现jdk动态代理需要实现类的接口类型
public interface IUserDao {
public void a();
public void b();
}
public class UserDaoimplements IUserDao{
@Override
public void a() {
System.out.println("aaa");
}
@Override
public void b() {
System.out.println("bbb");
}
}
/**
* 代理工厂
* 创建动态代理
* 动态代理不需要实现接口,但需要制定接口类型
*/
/**
* jdk 动态代理有一个限制,就是代理对象必须实现一个或多个接口 也叫 接口代理
* 如果想实现没有接口的类,就可以使用cglib 代理(子类代理)
*/
public class ProxyFactory {
private Objecttarget;
public ProxyFactory(Object target){
this.target = target;
}
/**
* 给目标对象生成代理对象
* @return
*/
public ObjectgetProxyInstance(){
return Proxy.newProxyInstance(
//类加载器
target.getClass().getClassLoader(),
//目标对象接口类型,使用泛型方式确认
target.getClass().getInterfaces(),
//触发事件处理器的方法,会把当前执行目标的对象的方法作为参数传入
new InvocationHandler() {
@Override
public Objectinvoke(Object proxy, Method method, Object[] args)throws Throwable {
System.out.println("开始事务");
Object returnValue = method.invoke(target,args);
System.out.println("提交事务");
return returnValue;
}
}
);
}
}
public class TestProxy {
public static void main(String[] args) {
UserDao target =new UserDao();
//原始的类型
System.out.println(target.getClass());
ProxyFactory proxyFactory =new ProxyFactory(target);
//必须指定接口类型,否则
//Exception in thread "main" java.lang.ClassCastException: com.sun.proxy.$Proxy0 cannot be cast to com.hbj.Test.proxy.UserDao
// at com.hbj.Test.proxy.TestProxy.main(TestProxy.java:11)
IUserDao proxyInstance = (IUserDao) proxyFactory.getProxyInstance();
System.out.println(proxyFactory.getClass());
proxyInstance.a();
}
}
❼ JDK是什么意思
JDK是Java语言的软件开发工具包,主要用于移动设备、嵌入式设备上的java应用程序。JDK是整个java开发的核心,它包含了JAVA的运行环境(JVM+Java系统类库)和JAVA工具。
没有JDK的话,无法编译Java程序(指java源码.java文件),如果想只运行Java程序(指class或jar或其它归档文件),要确保已安装相应的JRE。
(7)jdk代理是什么意思扩展阅读
JDK包含的基本组件包括:
javac编译器,将源程序转成字节码
jar打包工具,将相关的类文件打包成一个文件
javadoc文档生成器,从源码注释中提取文档
jdbdebugger,查错工具
java运行编译后的java程序(.class后缀的)
appletviewer:小程序浏览器,一种执行HTML文件上的Java小程序的Java浏览器。
参考资料来源:网络-jdk
❽ cglib动态代理和jdk动态代理的区别与应用
jdk动态代理:需要有顶层接口才能使用,但是在只有顶层接口的时候也可以使用,常见是mybatis的mapper文件是代理.
cglib动态代理:可以直接代理类,使用字节码技术,不能对 final类进行继承。使用了动态生成字节码技术。