Java代码解释静态代理和动态代理的区别

简介: ### 静态代理与动态代理简介**静态代理**:代理类在编译时已确定,目标对象和代理对象都实现同一接口。代理类包含对目标对象的引用,并在调用方法时添加额外操作。**动态代理**:利用Java反射机制在运行时生成代理类,更加灵活。通过`Proxy`类和`InvocationHandler`接口实现,无需提前知道接口的具体实现细节。示例代码展示了两种代理方式的实现,静态代理需要手动创建代理对象,而动态代理通过反射机制自动创建。

静态代理

在静态代理中,代理类是固定的,在编译时就已经确定了。

目标对象接口(TargetInterface)

public interface TargetInterface {
   
    void request();
}

目标对象实现类(TargetClass)

public class TargetClass implements TargetInterface {
   
    @Override
    public void request() {
   
        System.out.println("目标操作");
    }
}

代理对象接口(ProxyInterface)与代理对象实现类(ProxyClass)

public interface ProxyInterface extends TargetInterface {
   
    // 代理对象包含对目标对象的引用
    private TargetInterface target;

    public ProxyInterface(TargetInterface target) {
   
        this.target = target;
    }

    @Override
    public void request() {
   
        System.out.println("前置操作");
        target.request();
        System.out.println("后置操作");
    }
}

测试类(Test)

public class Test {
   
    public static void main(String[] args) {
   
        // 创建目标对象实例
        TargetInterface target = new TargetClass();
        // 创建代理对象实例
        ProxyInterface proxy = new ProxyClass(target);
        // 通过代理对象执行请求操作
        proxy.request();
    }
}

动态代理

动态代理使用Java的反射机制来创建代理类,可以在运行时生成。

目标对象接口(TargetInterface)

public interface TargetInterface {
   
    void request();
}

实现目标接口的目标对象实现类(TargetClass)

public class TargetClass implements TargetInterface {
   
    @Override
    public void request() {
   
        System.out.println("目标操作");
    }
}

使用动态代理创建代理对象

使用Java的Proxy类来创建动态代理:

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

public class DynamicProxyTest {
   
    public static void main(String[] args) {
   
        // 创建目标对象实例
        TargetInterface target = new TargetClass();

        // 定义InvocationHandler实现类
        InvocationHandler handler = new InvocationHandler() {
   
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
   
                System.out.println("前置操作");
                Object result = method.invoke(target, args);
                System.out.println("后置操作");
                return result;
            }
        };

        // 使用Proxy类创建代理对象
        TargetInterface proxyInstance = (TargetInterface) Proxy.newProxyInstance(
                TargetClass.class.getClassLoader(),
                new Class[]{
   TargetInterface.class},
                handler
        );

        // 通过代理对象执行请求操作
        proxyInstance.request();
    }
}

总结

  • 静态代理:在编译时就已经确定了代理类,较为固定。适用于功能简单的场景。
  • 动态代理:利用Java的反射机制可以在运行时生成代理类,更加灵活和通用。

通过上述代码示例可以看到,静态代理需要手动创建代理对象并定义好方法;而动态代理则不需要提前知道接口的具体实现细节,只需编写InvocationHandler逻辑即可。

相关文章
|
13天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
35 3
|
2月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
56 24
|
20天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
56 2
|
2月前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
83 14
|
1月前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
82 5
|
1月前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
57 5
|
1月前
|
Java
java中面向过程和面向对象区别?
java中面向过程和面向对象区别?
25 1
|
2月前
|
Java API 开发者
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
|
1月前
|
安全 Java API
Java中的Lambda表达式:简化代码的现代魔法
在Java 8的发布中,Lambda表达式的引入无疑是一场编程范式的革命。它不仅让代码变得更加简洁,还使得函数式编程在Java中成为可能。本文将深入探讨Lambda表达式如何改变我们编写和维护Java代码的方式,以及它是如何提升我们编码效率的。
|
2月前
|
Java
Java将OffsetDateTime格式化为 yyyy-MM-dd HH:mm:ss 如何写代码?
Java将OffsetDateTime格式化为 yyyy-MM-dd HH:mm:ss 如何写代码?
38 0