代理设计模式8--代理模式探索

一、定义

        代理模式:为目标对象提供一种代理以控制对目标对象的访问。在某些情况下,一个对象不适合或者不能直接引用目标对象(比如包访问权限的类),而代理对象可以在客户端和目标对象之间起到中介的作用。

        代理类就像秘书一样,为老板代理一些琐碎的事情,管事的还是老板。

   为目标对象创建一个代理,以控制对它的直接访问。

二、分类

        静态代理,JDK动态代理,Cglib动态代理。

三、静态代理

        优点:静态代理本质上就是创建一个类的包装类,所以不用修改被代理的对象。

        缺点:如果被代理的对象加了一个方法,那么代理类也要同步加一个方法。

        下面是一个简单的例子

//被代理的对象或者说是委托对象,实现了接口
public class Subject implements Operation{
    @Override
    public void function() {
        System.out.println("主体操作");
    }
}
//代理的对象,与委托对象实现了相同的接口
class Proxy implements Operation{
    //持有委托对象的引用
    private Subject subject;
    //使用构造函数接收
    public Proxy(Subject subject) {
        this.subject = subject;
    }

    @Override
    public void function() {
        System.out.println("增强操作,如 日志,安全,事务,缓存");
        subject.function();
        System.out.println("后置处理,如记录操作时间");
    }
}

四、JDK动态代理

        优点:JDK动态代理把代理方法的处理都集中到了InvocationHandler这个类的实现中,所以方便统一增加类似于切面的功能,比如记录时间,增加日志、缓存等。

        缺点:JDK动态代理只能代理接口的实现

        运行时:根据我们在Java代码中的“指示”,动态的生成了相关类

        下面是一个简单的例子

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

public class DynamicProxy implements InvocationHandler{
    //被代理的对象的引用,或者说是委托对象,也可以称它为真实对象
    private Object target;

    public DynamicProxy(Object target) {
        this.target = target;
    }

    public Object getInstance(){
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),
                target.getClass().getInterfaces(),this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //前置增强
        System.out.println("增强操作,如 日志,安全,事务,缓存");
        //调用被代理对象的业务方法
        Object result = method.invoke(target,args);
        System.out.println("后置增强,用于记录数据,或者记录时间,将数据存到全局map中等");
        return result;
    }

    public static void main(String[] args) {
        //业务实现类实现了Operatino接口
        Operation subject = new Subject();
        //创建动态代理对象
        DynamicProxy proxy = new DynamicProxy(subject);
        //创建代理实例对象
        Operation proxyInstance = (Operation)proxy.getInstance();
        //调用代理对象的方法,包裹了真实对象的方法
        proxyInstance.function();
    }
}

五、Cglib动态代理

        如果目标没有实现接口,那么可以用cglib动态代理。cglib是针对类来实现代理的,它生成目标类的一个子类作为代理类。

   Cglib动态代理的缺点就是不能对final修饰的类进行代理,final修饰的类不能生成子类。

        下面是一个简单的例子

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;

public class CglibProxy implements MethodInterceptor {
    //被代理的对象,或者是说是真实对象,委托对象
    private Object target;
    public CglibProxy(Object target) {
        this.target = target;
    }

    public Object getInstance() {
        //创建代理对象的工厂
        Enhancer enhancer = new Enhancer();
        //设置代理对象的父类是当前的真实对象
        enhancer.setSuperclass(this.target.getClass());
        // 回调方法,当调用代理对象的时候,让真实对象去操作
        enhancer.setCallback(this);
        // 创建代理对象,并返回
        return enhancer.create();
    }

    @Override
    public Object intercept(Object proxy, Method method, Object[] args, 
                         MethodProxy methodProxy) throws Throwable {
        //前置增强
        System.out.println("增强操作,如 日志,安全,事务,缓存");
        Object result = method.invoke(target,args);
        System.out.println("后置增强");
        return result;
    }

    public static void main(String[] args) {
        Temp temp = new Temp();
        CglibProxy proxy = new CglibProxy(temp);
        Temp tempInstance = (Temp)proxy.getInstance();
        tempInstance.login();
        /**
         * 运行结果:
         * 增强操作,如 日志,安全,事务,缓存
         * 临时登录
         * 后置增强,用于记录数据,或者记录时间,将数据存到全局map中等
         */
    }
}

六、增强工具类例子

        在类中添加抽象方法,让子类去继承它,实现其它增强的方法。这有点像拦截器的感觉了。重写invoke方法,父类的用super.invoke()方法回调。

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

public abstract class DynamicProxy implements InvocationHandler{
    //被代理的对象的引用,或者说是委托对象,也很称它为真实对象
    private Object target;

    public DynamicProxy(Object target) {
        this.target = target;
    }

    //前处理
    public abstract Object prehandle();
    //后处理
    public abstract Object posthandle();

    public Object getInstance(){
        /**
         * 参数1:类加载器
         * 参数2:被代理对象所实现的接口,我们动态生成的代理类也将实现这些接口
         * 参数3:使用哪个调用处理器的invoke方法,当前就使用这个类的实例对象的invoke方法
         *
         */
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),
                target.getClass().getInterfaces(),this);
    }

    /**
     *
     * @param proxy   生成的代理对象的引用
     * @param method  代理对象调用的方法
     * @param args    参数数组
     * @return        调用此方法的结果
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        prehandle();
        //调用被代理对象的业务方法
        Object result = method.invoke(target, args);
        posthandle();
        return result;
    }
}

深度好文:

        谈谈JAVA的代理模式认识 一——为什么使用代理模式

        java的动态代理机制详解

        代理模式(Proxy Pattern),静态代理 VS 动态代理

 

分割线--------------------------------------------------------------------------------------------

下一篇:代理设计模式9--使用AOP风格的代理模式

刻意练习

        (1)一句话描述代理设计模式

        (2)静态代理模式的优点与缺点

        (3)JDK动态代理的缺点

        (4)JDK动态代理的优点

        (5)一般代理Controller,为什么用Cglib代理,而不用Jdk动态代理

        (6) Cglib动态代理的一个比较常见的缺陷

《餐馆点餐管理系统——基于Java和MySQL的课程设计解析》 在信息技术日益发达的今天,餐饮行业的数字化管理已经成为一种趋势。本次课程设计的主题是“餐馆点餐管理系统”,它结合了编程语言Java和数据库管理系统MySQL,旨在帮助初学者理解如何构建一个实际的、具有基本功能的餐饮管理软件。下面,我们将深入探讨这个系统的实现细节及其所涉及的关键知识点。 我们要关注的是数据库设计。在“res_db.sql”文件中,我们可以看到数据库的结构,可能包括菜品表、订单表、顾客信息表等。在MySQL中,我们需要创建这些表格并定义相应的字段,如菜品ID、名称、价格、库存等。此外,还要设置主键、外键来保证数据的一致性和完整性。例如,菜品ID作为主键,确保每个菜品的唯一性;订单表中的顾客ID和菜品ID则作为外键,与顾客信息表和菜品表关联,形成数据间的联系。 接下来,我们来看Java部分。在这个系统中,Java主要负责前端界面的展示和后端逻辑的处理。使用Java Swing或JavaFX库可以创建用户友好的图形用户界面(GUI),让顾客能够方便地浏览菜单、下单。同时,Java还负责与MySQL数据库进行交互,通过JDBC(Java Database Connectivity)API实现数据的增删查改操作。在程序中,我们需要编写SQL语句,比如INSERT用于添加新的菜品信息,SELECT用于查询所有菜品,UPDATE用于更新菜品的价格,DELETE用于删除不再提供的菜品。 在系统设计中,我们还需要考虑一些关键功能的实现。例如,“新增菜品和价格”的功能,需要用户输入菜品信息,然后通过Java程序将这些信息存储到数据库中。在显示所有菜品的功能上,程序需要从数据库获取所有菜品数据,然后在界面上动态生成列表或者表格展示。同时,为了提高用户体验,可能还需要实现搜索和排序功能,允许用户根据菜品名称或价格进行筛选。 另外,安全性也是系统设计的重要一环。在连接数据库时,要避免SQL注入攻击,可以通过预编译的PreparedStatement对象来执行SQL命令。对于用户输入的数据,需要进行验证和过滤,防止非法字符和异常值。 这个“餐馆点餐管理系统”项目涵盖了Java编程、数据库设计与管理、用户界面设计等多个方面,是一个很好的学习实践平台。通过这个项目,初学者不仅可以提升编程技能,还能对数据库管理和软件工程有更深入的理解。在实际开发过程中,还会遇到调试、测试、优化等挑战,这些都是成长为专业开发者不可或缺的经验积累
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小大宇

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值