静态代理与动态代理_java静态代理动态代理

静态代理与动态代理_java静态代理动态代理讲解静态代理与动态代理的实现示例,静态代理与动态代理的区别

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

静态代理

静态代理使用场景

在实际使用中,经常会遇到一些场景:想在多方法前或方法后加一个共同的方法,如在方法执行前后打印日志、在方法执行完成发送消息通知等。这个时候,我们可以在方法前后调用公用方法,也可以使用代理实现,当然,代理的作用不止如此,这里不做赘述,实现方法如下:

实现方法

先创建一个接口

public interface TestInterface {

    void showMessage();
}

创建一个具体的类,实现这个接口,再创建一个代理类,也实现这个接口

代理类加一个有参构造方法,创建代理类对象的时候传具体类对象即可

// 具体类
public class TestImpl implements TestInterface {

    @Override
    public void showMessage() {
        System.out.println("message");
    }
}

// 代理类
class TestProxy implements TestInterface {

    private TestInterface testInterface;

    public TestProxy(TestInterface testInterface) {
        this.testInterface = testInterface;
    }

    @Override
    public void showMessage() {
        printLogBefore();
        testInterface.showMessage();
        printLogAfter();
    }

    private void printLogBefore() {
        System.out.println("logger before");
    }

    private void printLogAfter() {
        System.out.println("logger after");
    }
}

调用代理类

public static void main(String[] args) throws Exception {
        TestProxy t = new TestProxy(new TestImpl());
        t.showMessage();
    }

执行结果

logger before
message
logger after

JDK动态代理

静态代理的具体类与代理类的对应关系是一对一,那么动态代理可以理解为多对多

动态代理是通过反射实现的

实现方法:

 先创建一个接口

public interface TestInterface {

    void showMessage();
}

创建一个具体的类,实现这个接口,再创建一个代理类,这里与静态代理不同,动态代理的代理类实现的是java.lang.reflect.InvocationHandler接口

被代理类

public class TestImpl implements TestInterface {

    @Override
    public void showMessage() {
        System.out.println("show message");
    }
}

代理类

public class TestProxy<T> implements InvocationHandler {

    private T obj;

    public T build(T obj) {
        this.obj = obj;
        return (T) Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        printLogBefore();
        method.invoke(obj, args);
        printLogAfter();
        return null;
    }

    private void printLogBefore() {
        System.out.println("print log before");
    }

    private void printLogAfter() {
        System.out.println("print log after");
    }
}

调用方法

public static void main(String[] args) {
        TestInterface t = new TestProxy<TestInterface>().build(new TestImpl());
        t.showMessage();
    }

返回结果

print log before
show message
print log after

 静态代理与动态代理的区别

  • 静态代理是在编译时已经创建好的,动态代理是在程序运行过程中通过反射创建的
  • 静态代理在运行前就知道是代理哪个类,而动态代理需要在运行的时候才能确定
  • 静态代理通常只代理一个类,而动态代理可以代理接口下所有类
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/183834.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • JavaScript给元素添加class属性

    JavaScript给元素添加class属性注意:element.classList.remove()、element.classList.add()—ie9及以下不兼容//移除div的class属性obj.classList.remove(‘active’);//添加class属性值//方式一obj.className+=’newactive’;//方式二//obj.className=…

    2022年6月22日
    276
  • phantomjs环境搭建已经运行

    phantomjs环境搭建已经运行

    2021年12月6日
    42
  • ActiveMQ详细入门教程系列(一)

    ActiveMQ详细入门教程系列(一)

    2020年11月20日
    265
  • 谷歌的技术_探究GNSS技术在

    谷歌的技术_探究GNSS技术在文章目录引言TrueTime事务读写事务快照读只读事务总结引言Spanner是一个全球分布式的数据库,从数据模型来看Spanner很像BigTable,都是类似于key对应着一行数据,但是却并不一样,Spanner中衍生出了“目录”的概念(把两张表合并存储)。这并不是重点,Spanner的重是它是第一个在全球范围内传递数据且保证外部一致的分布式事务的系统,且支持几种特定的事务,这显然是一个很困难的问题,我们会在文章中加以描述,这篇文章主要对Spanner的事务以及实现事务所使用的TrueTimeAP

    2025年6月2日
    3
  • HBase面试题精讲「建议收藏」

    HBase面试题精讲「建议收藏」1.HBase的特点是什么?1)大:一个表可以有数十亿行,上百万列;2)无模式:每行都有一个可排序的主键和任意多的列,列可以根据需要动态的增加,同一张表中不同的行可以有截然不同的列;3)面向列:面向列(族)的存储和权限控制,列(族)独立检索;4)稀疏:空(null)列并不占用存储空间,表可以设计的非常稀疏;5)数据多版本:每个单元中的数据可以有多个版本,默认情况下版本号自动分配,是单元格插入时的时间戳;6)数据类型单一:Hbase中的数据都是字符串,没有类型。2.HBase和Hive

    2022年5月6日
    51
  • ThinkPHP模板IF标签用法详解

    ThinkPHP模板IF标签用法详解

    2021年9月18日
    57

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号