首页 > 编程语言 >Java基础--代理

Java基础--代理

时间:2024-09-05 14:25:41浏览次数:5  
标签:Java String -- void 代理 class public size

目录

一、代理的基本概念

举例:

二、代理模式的好处

三、代理模式怎么用?

1、 基于jdk实现的静态代理

注意:

静态代理举例

静态代理举例2销售店(多个核心类)

总结

2、基于jdk实现的动态代理

动态代理举例,销售店


一、代理的基本概念

代理是一种设计模式。我们使用代理对象来代替对真实对象的访问。

举例:

我们要设计一个银行的转账系统,转账时必然会涉及以下方法。
1、对转账金额的确认和对转账双方的身份的确认
2、转账方法(核心)
3、事后服务

假设我们通过支付宝进行支付

这里我们通过支付宝(代理类) 来代理银行(核心类)中的核心方法,这样就避免了许多方法在同一个类中造成功能过多,也可以避免用户直接接触核心方法。

二、代理模式的好处

1、控制访问:核心业务交给专门部门代理,不让用户直接访问核心业务。

2、功能增强:可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标对象的功能。

三、代理模式怎么用?

实现代理模式分为静态代理和动态代理两种方式。

其中动态代理又分为基于jdk实现的动态代理和基于CGLB实现的动态代理

1、 基于jdk实现的静态代理

注意:

1、代理类和核心类都必须实现同一个接口,接口在这里的作用是通知代理类所代理的核心是什么 。

 2、核心类也叫做目标类

静态代理举例

就以上面的转账系统为例

定义一个转账接口,里面有核心方法转账。

public interface zhuanzhang {
    public void zhuan(String A,String B,Double money);

}

银行类(核心类) 

public class yinhang implements zhuanzhang{
    @Override
    public void zhuan(String A, String B, Double money) {
        System.out.println(A+"向"+B+"转账了"+money+"元");
    }
}

支付宝类(代理类) 创建了核心对象,拓展了转账方法。

public class zhifubao implements zhuanzhang{
    private yinhang abank = new yinhang();
    private void yanzheng(){
        System.out.println("对A进行身份验证");
        System.out.println("对B进行身份验证");
        System.out.println("对金额进行验证");
    }
    @Override
    public void zhuan(String A, String B, Double money) {
        yanzheng();
        abank.zhuan(A,B,money);
        fuwu();
    }
    private void fuwu(){
        System.out.println("进行事后服务");
    }
}

测试类

public class test {
    public static void main(String[] args) {
        zhifubao a = new zhifubao();
        a.zhuan("zhangsan","lisi",200.0);
    }
}

 运行结果

 内存图解释

        main方法先创建zhifubao类的对象a,在a中创建了yinhang类的abank对象,运行时调用支付宝中的zhuanzhang方法(在这个转账方法中调用了yanzheng()和fuwu()以及abank的转账方法),通过这个方法调用abank的转账方法。


静态代理举例2销售店(多个核心类)

创建衣服和鞋子接口与核心类

public interface Clothes {
    public void BuyClothes(String size);
}
public class ClothesFactory implements Clothes{
    @Override
    public void BuyClothes(String size) {
        System.out.println("定制一件大小为"+size+"的衣服");
    }
}
public interface Shoes {
    public void BuyShoes(String size);
}
public class ShoesFactory implements Shoes{
    @Override
    public void BuyShoes(String size) {
        System.out.println("定制鞋子的大小为"+size);
    }
}

创建销售店,进行代理 

public class XSD implements Clothes,Shoes{
    //代理目标类
    private ClothesFactory clothesFactory=new ClothesFactory();
    private ShoesFactory shoesFactory=new ShoesFactory();

    private void fuwu(){
        System.out.println("销售店进行服务");
    }

    @Override
    public void BuyClothes(String size) {
        clothesFactory.BuyClothes(size);
        fuwu();
    }

    @Override
    public void BuyShoes(String size) {
        shoesFactory.BuyShoes(size);
        fuwu();
    }
}

 

总结

        通过上面的例子我们不难发现,静态代理中,我们对目标对象的每个方法的增强都是手动完成的,非常不灵活(比如接口一旦新增加方法,目标对象和代理对象都要进行修改)且麻烦(需要对每个目标类都单独写一个代理类)。实际应用场景非常非常少。如果要在不同的银行进行转账,也就是有多个核心类,就会很复杂并且会破坏一些性质。这时就需要用动态代理。

2、基于jdk实现的动态代理

代理对象一对一生成

动态代理举例,销售店

创建核心接口和核心类

public interface Clothes {
    public void BuyClothes(String size);
}
public interface Shoes {
    public void BuyShoes(String size);
}
public class ShoesFactory implements Shoes{
    @Override
    public void BuyShoes(String size) {
        System.out.println("定制鞋子的大小为"+size);
    }
}
public class ClothesFactory implements Clothes{
    @Override
    public void BuyClothes(String size) {
        System.out.println("定制一件大小为"+size+"的衣服");
    }
}

 创建动态的销售店

其中getProxyInstance()方法获取目标类当中的核心方法,

invoke()方法:继承接口InvocationHandler,重写方法,在施行核心方法的过程中需要执行invoke方法

public class DTXSD implements InvocationHandler{
    private Object object;  //目标类   见内存图1
    public DTXSD(Object o) {     
        object = o;
    }

    /**
     * 动态代理实现相关接口
     * 调用该方法你就知道了目标类当中的核心方法是什么
     * @return
     */
    public Object getProxyInstance(){  //object.getClass() 获取目标类的类对象
        return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(),this);
    }

    //三个参数的讲解
    //1.Object:jdk创建的代理类,无需赋值
    //2.Method:目标类当中的方法,jdk提供,无需赋值
    //3.Object[]:目标类当中的方法的参数,jdk提供,无需赋值
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        method.invoke(object,args);
        fuwu();
        return null;
    }
    private void fuwu(){
        System.out.println("销售店进行一系列的服务");
    }
}

测试类

public class Test {
    public static void main(String[] args) {

          ClothesFactory clothesFactory = new ClothesFactory();
          DTXSD dtxsd1 =  new DTXSD(clothesFactory);
          //已经知道了目标类当中的核心方法是什么
          Clothes clothes = (Clothes) dtxsd1.getProxyInstance();
          clothes.ByClothes("XXXL");
    }
}

运行结果

内存图(暂时)

public class XSD {
    private Object object; 
    public Shop(Object o){ 
        object = o;
    }
}

public class Test {
    public static void main(String[] args) {
        ClothesFactory clothesFactory = new ClothesFactory(); 
        XSD dtxsd = new XSD(clothesFactory);
    }
}

 

注意:在最后调用获取到的核心方法时必须要用clothes类调用,直接创建clothesfactory对象调用核心方法是没有用代理的。

标签:Java,String,--,void,代理,class,public,size
From: https://blog.csdn.net/a13641376265/article/details/141927641

相关文章

  • Java中的服务端点监控:Actuator与Micrometer
    Java中的服务端点监控:Actuator与Micrometer大家好,我是微赚淘客返利系统3.0的小编,是个冬天不穿秋裤,天冷也要风度的程序猿!今天我们将探讨如何在Java应用中实现服务端点监控,重点介绍SpringBootActuator和Micrometer这两个工具。通过示例代码,我们将展示如何配置和使用这些工具来监......
  • 你知道手机号也有风险画像吗?
        我们每个人或许拥有不止一个手机号,它们连接着我们的社交媒体,支付账号以及购物出行息息相关的方方面面。而且有这么一群人利用手机号操作进行着刀尖上舔血的半违法或违法活动。例如垃圾注册,短信轰炸,黄牛,薅羊毛等行为。于是手机号风险画像应运而生。也许我们都听过IP风险......
  • 最新版Kubernetes部署教程v1.31.0
    最新版Kubernetes高可用部署教程v1.31.0系统:Almalinux9架构:192.168.100.10control-plane-endpoint.k8s.localcontrol-plane-endpoint#负载均衡服务器,配置负载均衡后修改ip后续会讲解192.168.100.10masterA.k8s.localmasterA192.168.100.20masterB.k8s.localma......
  • Java应用的数据库连接池故障注入测试
    Java应用的数据库连接池故障注入测试大家好,我是微赚淘客返利系统3.0的小编,是个冬天不穿秋裤,天冷也要风度的程序猿!今天我们将探讨如何在Java应用中进行数据库连接池故障注入测试。这种测试帮助我们确保应用在面对数据库连接池故障时的稳定性和鲁棒性。我们将使用HikariCP作为连接......
  • 在模板中使用 Django 会话
    在Django中使用会话(session)可以让你在用户访问网站的过程中存储和访问临时数据。我们可以利用会话在速度计算器的例子中存储和显示上次计算的结果。1、问题背景在Django中,可以使用会话来存储用户数据。在某些情况下,我们需要在模板中使用会话数据。但是,在某些情况下,我们无法直接......
  • Java中的分布式任务调度:Quartz与Spring Task
    Java中的分布式任务调度:Quartz与SpringTask大家好,我是微赚淘客返利系统3.0的小编,是个冬天不穿秋裤,天冷也要风度的程序猿!今天我们将探讨如何在Java应用中实现分布式任务调度,重点介绍Quartz和SpringTask这两种常见的调度解决方案。我们将分别介绍这两种工具的基本用法,并演示如何......
  • 3292. 称检测点查询 来源:第二十次CCF-CSP计算机软件能力认证 枚举 排序
    #include<iostream>#include<cstdio>#include<algorithm>usingnamespacestd;constintN=210;pair<int,int>p[N];intn,X,Y;intmain(){cin>>n>>X>>Y;for(inti=1;i<=n;i++){......
  • AI大模型的硬件协同优化
    人工智能大模型已经成为当今科技领域的关键驱动力,它们不仅在语言理解、图像识别等任务中展现了非凡的能力,还在各个领域推动了科学研究和工业应用的进步。然而,这些大模型的训练和推理需求极高的计算资源,这促使硬件设计和优化成为实现其高效运行的关键因素之一。1.背景介绍随着深度......
  • Java中的服务端点安全性:Spring Security的高级特性
    Java中的服务端点安全性:SpringSecurity的高级特性大家好,我是微赚淘客返利系统3.0的小编,是个冬天不穿秋裤,天冷也要风度的程序猿!今天,我们将深入探讨SpringSecurity的高级特性,以增强Java应用的服务端点安全性。本文将展示如何利用SpringSecurity的强大功能来保护服务端点,涵盖配置......
  • 聊一下Jetpack AppStartUp的使用和原理。
    AppStartup是AndroidJetpack中的一个库,用于在应用启动时初始化组件,能简化启动序列并显式设置初始化依赖顺序,从而提高应用的启动速度。以下是关于AppStartup的使用和原理的介绍:使用方法添加依赖:在项目的模块级build.gradle文件中添加对AppStartup的依赖。implementatio......