享元和代理模式
文章目录
- 享元模式
- 1.引出享元模式
- 1.展示网站项目需求
- 2.传统方案解决
- 3.问题分析
- 2.享元模式
- 1.基本介绍
- 2.原理类图
- 3.外部状态和内部状态
- 4.类图
- 5.代码实现
- 1.AbsWebSite.java 抽象的网站
- 2.ConcreteWebSite.java 具体的网站,type属性是内部状态
- 3.WebSiteFactory.java 网站工厂,根据类型,获取对应类型的网站
- 4.User.java 使用网站的用户,是外部状态
- 5.Client.java
- 6.结果
- 6.小结
- 3.享元模式在Integer的应用
- 代理模式
- 1.基本介绍
- 1.介绍
- 2.简单类图
- 2.静态代理
- 1.基本介绍
- 2.类图
- 3.代码实现
- 1.目录结构
- 2.ITeacherDao.java 被代理的接口
- 3.TeacherDao.java 被代理的实现类
- 4.TeacherDaoProxy.java 静态代理类
- 5.Client.java
- 6.结果
- 2.动态代理(JDK代理)
- 1.基本介绍
- 2.类图
- 3.代码实现
- 1.ITeacherDao.java 被代理的接口
- 2.TeacherDao.java 被代理的类
- 3.ProxyFactory.java 代理工厂,返回动态代理对象
- 4.Client.java
- 5.结果
- 3.Cglib代理
- 1.基本介绍
- 2.注意事项
- 3.类图
- 4.引入四个jar包
- 5.代码实现
- 1.目录结构
- 2.TeacherDao.java 被代理的类
- 3.ProxyFactory.java 返回代理对象的工厂
- 4.Client.java
- 5.结果
- 6.几种变体
享元模式
1.引出享元模式
1.展示网站项目需求
2.传统方案解决
3.问题分析
2.享元模式
1.基本介绍
2.原理类图
3.外部状态和内部状态
4.类图
5.代码实现
1.AbsWebSite.java 抽象的网站
package com.sun;/*** Description: 抽象的网站* @Author sun* @Create 2024/6/6 19:45* @Version 1.0*/
public abstract class AbsWebSite {public abstract void use(User user);
}
2.ConcreteWebSite.java 具体的网站,type属性是内部状态
package com.sun;/*** Description: 具体的网站* @Author sun* @Create 2024/6/6 19:46* @Version 1.0*/
public class ConcreteWebSite extends AbsWebSite {// 网站的发布类型private String type = "";// 在创建网站时把具体的类型传进来public ConcreteWebSite(String type) {this.type = type;}@Overridepublic void use(User user) {System.out.println("网站的发布类型为:" + type + user.getName() + "在使用中");}
}
3.WebSiteFactory.java 网站工厂,根据类型,获取对应类型的网站
package com.sun;import java.util.HashMap;/*** Description: 网站的工厂类,根据需求,返回一个具体的网站* @Author sun* @Create 2024/6/6 19:49* @Version 1.0*/
public class WebSiteFactory {// 一个集合,充当池的作用private HashMap<String, ConcreteWebSite> pool = new HashMap<>();public AbsWebSite getWebsiteCategory(String type) {// 如果池中没有包含对应类型的对象,就创建一个对象,放到池中if (!pool.containsKey(type)) {pool.put(type, new ConcreteWebSite(type));}// 只要到这里了就必然可以拿到对象return (AbsWebSite) pool.get(type);}// 获取池中的网站总数public int getWebSiteCount() {return pool.size();}
}
4.User.java 使用网站的用户,是外部状态
package com.sun;/*** Description: 享元模式中的外部状态* @Author sun* @Create 2024/6/6 20:12* @Version 1.0*/
public class User {private String name;public User(String name) {this.name = name;}public String getName() {return name;}public void setName(String name) {this.name = name;}
}
5.Client.java
package com.sun;/*** Description:* @Author sun* @Create 2024/6/6 20:02* @Version 1.0*/
public class Client {public static void main(String[] args) {// 创建一个工厂类WebSiteFactory webSiteFactory = new WebSiteFactory();// 要一个以新闻形式发布的网站AbsWebSite webSite = webSiteFactory.getWebsiteCategory("新闻");webSite.use(new User("tom"));// 要一个以博客形式发布的网站,后面即使要更多相同类型的网站,都会直接从池中获取,而不会创建新的实例AbsWebSite webSite2 = webSiteFactory.getWebsiteCategory("博客");webSite2.use(new User("jack"));AbsWebSite webSite3 = webSiteFactory.getWebsiteCategory("博客");webSite3.use(new User("lisa"));AbsWebSite webSite4 = webSiteFactory.getWebsiteCategory("博客");webSite4.use(new User("milan"));int webSiteCount = webSiteFactory.getWebSiteCount();System.out.println("webSiteCount = " + webSiteCount);}
}
6.结果
6.小结
3.享元模式在Integer的应用
代理模式
1.基本介绍
1.介绍
2.简单类图
2.静态代理
1.基本介绍
2.类图
3.代码实现
1.目录结构
2.ITeacherDao.java 被代理的接口
package com.sun.staticproxy;/*** Description: 接口* @Author sun* @Create 2024/6/7 19:05* @Version 1.0*/
public interface ITeacherDao {void teach();
}
3.TeacherDao.java 被代理的实现类
package com.sun.staticproxy;/**TeacherDaoProxy.java* Description: 实现类* @Author sun* @Create 2024/6/7 19:06* @Version 1.0*/
public class TeacherDao implements ITeacherDao{@Overridepublic void teach() {System.out.println("老师正在授课中");}
}
4.TeacherDaoProxy.java 静态代理类
package com.sun.staticproxy;/*** Description: 代理TeacherDao* @Author sun* @Create 2024/6/7 19:06* @Version 1.0*/
public class TeacherDaoProxy implements ITeacherDao{// 使用构造器进行聚合private ITeacherDao teacherDao;public TeacherDaoProxy(ITeacherDao teacherDao) {this.teacherDao = teacherDao;}@Overridepublic void teach() {System.out.println("代理开始");teacherDao.teach();System.out.println("代理结束");}
}
5.Client.java
package com.sun.staticproxy;/*** Description:* @Author sun* @Create 2024/6/7 19:11* @Version 1.0*/
public class Client {public static void main(String[] args) {TeacherDaoProxy teacherDaoProxy = new TeacherDaoProxy(new TeacherDao());teacherDaoProxy.teach();}
}
6.结果
2.动态代理(JDK代理)
1.基本介绍
2.类图
3.代码实现
1.ITeacherDao.java 被代理的接口
package com.sun.dynamic;/*** Description: 被代理的接口* @Author sun* @Create 2024/6/7 19:25* @Version 1.0*/
public interface ITeacherDao {void teach(String name);}
2.TeacherDao.java 被代理的类
package com.sun.dynamic;/*** Description: 实现类* @Author sun* @Create 2024/6/7 19:06* @Version 1.0*/
public class TeacherDao implements ITeacherDao {@Overridepublic void teach(String name) {System.out.println(name + "老师正在授课中");}}
3.ProxyFactory.java 代理工厂,返回动态代理对象
package com.sun.dynamic;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;/*** Description: 动态代理对象,可以对其代理的对象的任意方法添加任意操作* @Author sun* @Create 2024/6/7 19:27* @Version 1.0*/
public class ProxyFactory {// 构造器聚合一个目标对象private Object target;public ProxyFactory(Object target) {this.target = target;}// 给目标对象,生成一个代理对象public Object getProxyInstance() {/*** 参数说明:* ClassLoader loader:指定当前目标对象使用的类加载器* Class<?>[] interfaces:是目标对象实现的接口类型,使用泛型方法确认类型* InvocationHandler h:是事件处理,当使用代理对象调用目标对象的方法时会触发*/return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), new InvocationHandler() {/*** method:目标方法的Method对象,可以用来调用目标任何方法* args:目标方法的参数,也就是动态代理对象调用目标任何方法时传入的参数* @return 返回调用目标方法的返回值,也可以返回null* @throws Throwable*/@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("method = " + method + ", args = " + Arrays.toString(args));System.out.println("动态代理开始");/*invoke方法的第一个参数是目标对象,第二个参数的目标方法的参数*/Object result = method.invoke(target, args);System.out.println("动态代理结束");return result;}});}
}
4.Client.java
package com.sun.dynamic;/*** Description:* @Author sun* @Create 2024/6/7 19:47* @Version 1.0*/
public class Client {public static void main(String[] args) {// 创建一个目标对象ITeacherDao target = new TeacherDao();// 得到代理对象ITeacherDao proxy = (ITeacherDao) new ProxyFactory(target).getProxyInstance();// 使用代理对象调用方法proxy.teach("李华");}
}
5.结果
3.Cglib代理
1.基本介绍
2.注意事项
3.类图
4.引入四个jar包
5.代码实现
1.目录结构
2.TeacherDao.java 被代理的类
package com.sun.cglib;/*** Description: 被代理的类* @Author sun* @Create 2024/6/8 19:37* @Version 1.0*/
public class TeacherDao {public void teach(String name) {System.out.println(name + "老师授课中,使用的是cglib代理,不需要实现接口");}
}
3.ProxyFactory.java 返回代理对象的工厂
package com.sun.cglib;import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;/*** Description:* @Author sun* @Create 2024/6/8 19:38* @Version 1.0*/
public class ProxyFactory implements MethodInterceptor {// 构造器聚合一个目标对象private Object target;public ProxyFactory(Object target) {this.target = target;}// 返回目标对象的代理对象public Object getProxyInstance() {//1.创建一个工具类Enhancer enhancer = new Enhancer();//2.设置父类enhancer.setSuperclass(target.getClass());//3.设置回调函数enhancer.setCallback(this);//4.创建子类对象return enhancer.create();}/*** 当使用代理对象调用目标对象的函数时,就会跳到这个函数,跟之前动态代理时类似* @param o* @param method 代理对象调用的目标对象的函数* @param args 函数的参数* @param methodProxy* @return* @throws Throwable*/@Overridepublic Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {System.out.println("cglib代理模式开始");// 调用目标对象的函数System.out.println("method = " + method);System.out.println("args = " + args);Object returnVal = method.invoke(target, args);System.out.println("cglib代理结束");return returnVal;}
}
4.Client.java
package com.sun.cglib;/*** Description:* @Author sun* @Create 2024/6/8 19:50* @Version 1.0*/
public class Client {public static void main(String[] args) {// 创建一个目标对象TeacherDao teacherDao = new TeacherDao();// 得到目标对象的代理对象ProxyFactory proxyFactory = new ProxyFactory(teacherDao);TeacherDao proxyInstance = (TeacherDao) proxyFactory.getProxyInstance();// 使用代理对象来调用目标对象函数,则会被代理proxyInstance.teach("小明");}
}
5.结果
6.几种变体
相关文章:

享元和代理模式
文章目录 享元模式1.引出享元模式1.展示网站项目需求2.传统方案解决3.问题分析 2.享元模式1.基本介绍2.原理类图3.外部状态和内部状态4.类图5.代码实现1.AbsWebSite.java 抽象的网站2.ConcreteWebSite.java 具体的网站,type属性是内部状态3.WebSiteFactory.java 网站…...
[英语单词] ellipsize,动词化后缀 -ize
openvswitch manual里的一句话:里面有使用ellipsize,但是查字典是没有这个单词,这就是创造出来的动词。将单词ellipsis,加动词化后缀,-ize。 Often we ellipsize arguments not important to the discussion, e.g.: &…...
自然资源-测绘地信专业术语,值得收藏!
自然资源-测绘地信专业术语,值得收藏! 1、1954年北京坐标系 1954年我国决定采用的国家大地坐标系,实质上是由原苏联普尔科沃为原点的1942年坐标系的延伸。 2、1956年黄海高程系统 根据青岛验潮站1950年一1956年的验潮资料计算确定的平均海面…...
如何在小程序中实现页面之间的返回
在小程序中实现页面之间的返回,通常有以下几种方法,这些方法各有特点,适用于不同的场景: 1. 使用wx.navigateBack方法 描述:wx.navigateBack是微信小程序中用于关闭当前页面,返回上一页面或多级页面的API…...
深入解析数据结构之B树:平衡树中的王者
在计算机科学中,数据结构是算法和程序设计的基础。而在众多数据结构中,B树作为一种平衡树,在数据库和文件系统中有着广泛应用。本文将详细介绍B树的概念、特点、操作、优缺点及其应用场景,帮助读者深入理解这一重要的数据结构。 …...

18. 第十八章 继承
18. 继承 和面向对象编程最常相关的语言特性就是继承(inheritance). 继承值得是根据一个现有的类型, 定义一个修改版本的新类的能力. 本章中我会使用几个类来表达扑克牌, 牌组以及扑克牌性, 用于展示继承特性.如果你不玩扑克, 可以在http://wikipedia.org/wiki/Poker里阅读相关…...
OperationalError: (_mysql_exceptions.OperationalError)
OperationalError: (_mysql_exceptions.OperationalError) (2006, MySQL server has gone away) 这个错误通常表示客户端(例如你的 Python 程序使用 SQLAlchemy 连接到 MySQL 数据库)和 MySQL 服务器之间的连接被异常关闭了。这个问题可能由多种原因引起,以下是一些常见的原…...
DocGraph相关概念
结合简化版的直观性和专业版的深度,我们可以得到一个既易于理解又包含专业细节的DocGraph概念讲解。 DocGraph概述(简化版) 想象DocGraph就像是文章信息的地图。它通过拆分文档、识别关键词、分析关系,并最终以图形方式呈现这些…...

MySQL限制登陆失败次数配置
目录 一、限制登陆策略 1、Windows 2、Linux 一、限制登陆策略 1、Windows 1)安装插件 登录MySQL数据库 mysql -u root -p 执行命令安装插件 #限制登陆失败次数插件 install plugin CONNECTION_CONTROL soname connection_control.dll;install plugin CO…...
洛谷题解 - P1192 台阶问题
目录 题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示代码 题目描述 有 N N N 级台阶,你一开始在底部,每次可以向上迈 1 ∼ K 1\sim K 1∼K 级台阶,问到达第 N N N 级台阶有多少种不同方式。 输入格式 两个正整数 N , K …...

Unity贪吃蛇改编【详细版】
Big and small greedy snakes 游戏概述 游戏亮点 通过对称的美感,设置两条贪吃蛇吧,其中一条加倍成长以及加倍减少,另一条正常成长以及减少,最终实现两条蛇对整个界面的霸占效果。 过程中不断记录两条蛇的得分情况,…...
React中数据响应式原理
React作为当下最流行的前端框架之一,以其声明式编程和组件化架构而广受开发者喜爱。而React的数据响应式原理,是其高效更新DOM的核心机制。本文将深入探讨React中数据响应式原理,并结合代码示例进行论证。 响应式原理概述 在React中&#x…...

【FreeRTOS】ARM架构汇编实例
目录 ARM架构简明教程1. ARM架构电脑的组成1.2 RISC1.2 提出问题1.3 CPU内部寄存器1.4 汇编指令 2. C函数的反汇编 学习视频 【FreeRTOS入门与工程实践 --由浅入深带你学习FreeRTOS(FreeRTOS教程 基于STM32,以实际项目为导向)】 https://www.…...

【Linux】常见指令的使用
文章目录 which指令stat 指令wc指令echo指令tree 指令whoami指令clear指令alias指令ls指令pwd指令cd 指令touch指令mkdir指令(重要)rmdir指令 && rm 指令(重要)man指令(重要)cp指令(重要…...
C#面:详细阐述什么是 DTO
DTO(Data Transfer Object)是一种设计模式,用于在不同层之间传输数据。它的主要目的是在应用程序的不同部分之间传递数据,而不是直接传递实体对象。DTO通常是一个简单的POCO(Plain Old CLR Object)…...

「TCP 重要机制」三次握手四次挥手
🎇个人主页:Ice_Sugar_7 🎇所属专栏:计网 🎇欢迎点赞收藏加关注哦! 三次握手&四次挥手 🍉连接管理🍌三次握手🍌意义🍌四次挥手🍌TCP 状态转换…...

Java数据库编程
引言 在现代应用开发中,与数据库交互是不可或缺的一部分。Java提供了JDBC(Java Database Connectivity) API,允许开发者方便地连接到数据库并执行SQL操作。本文将详细介绍Java数据库编程的基础知识,包括JDBC的基本概念…...
决策树算法介绍:原理与案例实现
一、引言 决策树是一种常用于分类和回归任务的机器学习算法,因其易于理解和解释的特点,在数据分析和挖掘领域有着广泛应用。本文将介绍决策树算法的基本原理,并通过一个具体案例展示如何实现和应用该算法。 二、决策树算法原理 1. 决策树结…...
业务代表模式
业务代表模式 引言 在软件工程中,设计模式是解决常见问题的经典解决方案。它们为开发人员提供了一种方法,以优雅和可重用的方式处理软件开发中的挑战。业务代表模式(Business Delegate Pattern)是一种行为设计模式,它主要关注于将业务逻辑与表示层(如用户界面)分离,以…...

LeetCode 算法:反转链表 c++
原题链接🔗:反转链表 难度:简单⭐️ 题目 给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。 示例 1: 输入:head [1,2,3,4,5] 输出:[5,4,3,2,1] 示例 2:…...

铭豹扩展坞 USB转网口 突然无法识别解决方法
当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…...

《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...

定时器任务——若依源码分析
分析util包下面的工具类schedule utils: ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类,封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz,先构建任务的 JobD…...
测试markdown--肇兴
day1: 1、去程:7:04 --11:32高铁 高铁右转上售票大厅2楼,穿过候车厅下一楼,上大巴车 ¥10/人 **2、到达:**12点多到达寨子,买门票,美团/抖音:¥78人 3、中饭&a…...

新能源汽车智慧充电桩管理方案:新能源充电桩散热问题及消防安全监管方案
随着新能源汽车的快速普及,充电桩作为核心配套设施,其安全性与可靠性备受关注。然而,在高温、高负荷运行环境下,充电桩的散热问题与消防安全隐患日益凸显,成为制约行业发展的关键瓶颈。 如何通过智慧化管理手段优化散…...
TRS收益互换:跨境资本流动的金融创新工具与系统化解决方案
一、TRS收益互换的本质与业务逻辑 (一)概念解析 TRS(Total Return Swap)收益互换是一种金融衍生工具,指交易双方约定在未来一定期限内,基于特定资产或指数的表现进行现金流交换的协议。其核心特征包括&am…...
Spring AI与Spring Modulith核心技术解析
Spring AI核心架构解析 Spring AI(https://spring.io/projects/spring-ai)作为Spring生态中的AI集成框架,其核心设计理念是通过模块化架构降低AI应用的开发复杂度。与Python生态中的LangChain/LlamaIndex等工具类似,但特别为多语…...
云原生周刊:k0s 成为 CNCF 沙箱项目
开源项目推荐 HAMi HAMi(原名 k8s‑vGPU‑scheduler)是一款 CNCF Sandbox 级别的开源 K8s 中间件,通过虚拟化 GPU/NPU 等异构设备并支持内存、计算核心时间片隔离及共享调度,为容器提供统一接口,实现细粒度资源配额…...

负载均衡器》》LVS、Nginx、HAproxy 区别
虚拟主机 先4,后7...