java动态代理实现接口(基于接口的动态代理)

网友投稿 486 2023-01-05


本篇文章给大家谈谈java动态代理实现接口,以及基于接口的动态代理对应的知识点,希望对各位有所帮助,不要忘了收藏本站喔。 今天给各位分享java动态代理实现接口的知识,其中也会对基于接口的动态代理进行解释,如果能碰巧解决你现在面临的问题,别忘了关注本站,现在开始吧!

本文目录一览:

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();

    }

}

JAVA动态代理设计原理及如何实现

Java动态代理机制的出现,使得Java开发人员不用手工编写代理类,只要简单地制定一组接口及委托类对象,便能动态地获得代理类。代理类会负责将所有的方法调用分配到委托对象上反射执行,配置执行过程中,开发人员还可以进行修改

代理设计模式

代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问。代理类负责为委托类预处理消息、过滤消息并转发消息,以及进行消息被委托类执行后的后续处理。

为了保持行为的一致性,代理类和委托类通常会实现相同的接口

2. 引入代理能够控制对委托对象的直接访问,可以很好的隐藏和保护委托对象,也更加具有灵活性

代理机制及其特点

首先让我们来了解一下如何使用 Java 动态代理。具体有如下四步骤:

通过实现 InvocationHandler 接口创建自己的调用处理器;

通过为 Proxy 类指定 ClassLoader 对象和一组 interface 来创建动态代理类;

通过反射机制获得动态代理类的构造函数,其唯一参数类型是调用处理器接口类型;

通过构造函数创建动态代理类实例,构造时调用处理器对象作为参数被传入。

代理类实例的一些特点 

每个实例都会关联一个InvocationHandler(调用处理器对象),在代理类实例上调用其代理接口中声明的方法时,最终都会由InvocationHandler的invoke方法执行;

java.lang.Object中有三个方法也同样会被分派到调用处理器的 invoke 方法执行,它们是 hashCode,equals 和 toString;

代码示例

最后以一个简单的动态代理例子结束

Java 几种动态代理实现及其性能比较

1. 动态代理是指在运行时java动态代理实现接口,动态生成代理类。代理类的字节码将在运行时生成并载入当前的ClassLoader.
生成动态代理类的方法很多java动态代理实现接口,如JDK自带的动态代理、CGLIB、Javassist或者ASM库。
JDK动态代理使用简单,它内置在JDK中,因此不需要引入第三方Jar包,但相对功能比较弱。CGLIB和Javassist都是高级的字节码生成库,总体性能比JDK自带的动态代理好,而且功能十分强大。ASM是低级的字节码生成工具,使用ASM已经近乎在于使用Javabytecode编程,对开发人员要求较高,也是性能最好的一种动态代理生辰工具。但ASM的使用是在过于繁琐,而且性能也没有数量级的提升,与CGLIB等高级字节码生成工具相比,ASM程序的可维护性也较差。
JDK实现
1、步骤
1)通过实现InvocationHandler接口创建自己的调用处理器
2)通过为Proxy类指定ClassLoader对象和一组interface来创建动态代理类
3)通过反射机制获得动态代理类的构造函数,其唯一参数类型是调用处理器接口类型
4)通过构造函数创建动态代理类实例,构造时调用处理器对象作为参数被传入
2、创建代理
//InvocationHandlerImpl 实现java动态代理实现接口了InvocationHandler接口,并能实现方法调用从代理类到委托类的分派转发
//其内部通常包含指向委托类实例的引用,用于真正执行分派转发过来的方法调用
InvocationHandler handler = new InvocaitonHandlerImpl(..);
//通过Proxy为包括Interface接口在内的一组接口动态创建代理类的对象
Class clazz = Proxy.getProxyClass(classLoader,new Class[]{Interface.class,...});
//通过反射从生成的类对象获得构造函数对象
Constructor constructor = clazz.getConstructor(new Class[]{InvocationHandler.class});
//通过构造函数对象创建动态代理类实例
Interface Proxy = (Interface)constructor.newInstance(new Object[]{handler});
//Proxy类的静态方法newProxyInstance对上面具体步骤的后三步做了封装,简化了动态代理对象的获取过程。
//InvocationHandlerImpl实现了InvocaitonHandler接口,并能实现方法调用从代理类到委托类的分派转发
InvocaitonHandler handler = new InvocationHandlerImpl(..);
//通过Proxy直接创建动态代理类实例
nterface proxy = (Interface)Proxy.newProxyInstance(classLoader,new Class[]{Interface.class},handler);
3、代码
/**
* 接口
*
*/
public interface IDBQuery {
String request();
}
/**
* 真实的实现类,具体的目标对象
*
*/
public class DBQuery implements IDBQuery {
public DBQuery(){
try {
Thread.sleep(1000); //可能包含数据库连接等耗时操作
} catch (InterruptedException e) {
e.printStackTrace();
}
}
@Override
public String request() {
return "request string";
}
}
/**
* JDK动态代理的实现类
*
*/
public class JdkDbQueryHandler implements InvocationHandler{
IDBQuery real = null; //主题接口
/**
* 生成Handler
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
if(real == null)
real = new DBQuery(); //如果是第一次调用,则生成真实对象
return real.request(); //使用真实主题完成实际的操作
}
/**
* 利用Handler生成动态代理对象
*/
public static IDBQuery createJdkProxy(){
//根据指定的类加载器和接口以及截获器,返回代理类的一个实例对象
//ClassLoader loader :指定被代理对象的类加载器
//Class[] Interfaces java动态代理实现接口:指定被代理对象所以事项的接口
//InvocationHandler h :指定需要调用的InvocationHandler对象
IDBQuery jdkProxy = (IDBQuery) Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), new Class[]{IDBQuery.class}, new JdkDbQueryHandler());
return jdkProxy;
}
}

关于java动态代理实现接口和基于接口的动态代理的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。 java动态代理实现接口的介绍就聊到这里吧,感谢你花时间阅读本站内容,更多关于基于接口的动态代理、java动态代理实现接口的信息别忘了在本站进行查找喔。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:做接口测试流程(接口测试做什么)
下一篇:详解SpringBoot+SpringSecurity+jwt整合及初体验
相关文章

 发表评论

暂时没有评论,来抢沙发吧~