享元和代理模式
文章目录
- 享元模式
- 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:…...
LIBPNG深度解析:构建企业级PNG处理架构的技术决策指南
LIBPNG深度解析:构建企业级PNG处理架构的技术决策指南 【免费下载链接】libpng LIBPNG: Portable Network Graphics support, official libpng repository 项目地址: https://gitcode.com/gh_mirrors/li/libpng LIBPNG作为PNG格式的官方参考实现库࿰…...
OpenClaw对话式编程:Qwen3.5-9B解释代码与生成可执行脚本
OpenClaw对话式编程:Qwen3.5-9B解释代码与生成可执行脚本 1. 为什么需要对话式编程助手? 作为一个经常需要写脚本处理数据的开发者,我发现自己80%的时间都花在重复性工作上:查文档、调试语法错误、验证代码逻辑。直到尝试用Open…...
颈腰椎病引发 “耳后疼痛”:耳根刺痛,可能是颈椎在 “捣乱”
很多人出现耳后持续性刺痛或按压痛,会误以为是中耳炎、腮腺炎,实则部分耳后疼痛与颈椎病变相关。颈椎病变压迫枕大神经(从颈椎延伸至耳后),会导致神经分布区域疼痛;同时颈椎肌肉痉挛、僵硬,牵拉…...
Wan2.1-umt5辅助数学公式处理:从图片或LaTeX中理解与转换数学表达式
Wan2.1-umt5辅助数学公式处理:从图片或LaTeX中理解与转换数学表达式 如果你在科研、教育或者出版行业工作过,一定遇到过这样的烦恼:看到一篇论文里的复杂公式,想把它录入到自己的文档里,只能一个字一个字地对着敲&…...
PX4飞控系统深度解析:从模块化架构到自主飞行核心技术揭秘
PX4飞控系统深度解析:从模块化架构到自主飞行核心技术揭秘 【免费下载链接】PX4-Autopilot PX4 Autopilot Software 项目地址: https://gitcode.com/gh_mirrors/px/PX4-Autopilot 你是否曾好奇,一个开源飞控系统如何支撑从微型无人机到工业级无人…...
如何通过Universal Android Debloater实现Android设备深度优化
如何通过Universal Android Debloater实现Android设备深度优化 【免费下载链接】universal-android-debloater Cross-platform GUI written in Rust using ADB to debloat non-rooted android devices. Improve your privacy, the security and battery life of your device. …...
B端拓客号码核验行业:痛点剖析、技术突围与发展思考氪迹科技法人 号码筛选系统,阶梯式价格
B端拓客的效率与质量,很大程度上取决于核心决策人触达的精准度,而企业法人、股东、董监高等群体的有效联系方式,正是打通这一环节的关键。作为拓客工作的前置基础性步骤,号码核验的质量直接关联拓客投入的回报效率,更是…...
2026Agent元年!手把手教你从0到1搭建高能智能体,小白也能秒变大神!
逼自己练完这些,你的Agent搭建就很牛了!!2026年可谓是Agent元年,智能体(AI Agent)正以惊人的速度重塑我们的工作方式,从简单的被动响应工具,进化为能自主规划、执行、协作的"数…...
CPU工作原理:从二进制加法器到计算系统
CPU工作原理:从二进制加法器到计算系统的演进 1. 计算需求与二进制表示 在数字计算领域,加法是最基础也是最重要的运算之一。让我们从一个简单的数学问题开始:6324 244675 ?这个看似简单的加法问题,揭示了计算系统的…...
告别重复造轮子,用快马AI一键生成高复用登录组件提升效率
在开发官网登录入口时,我们常常需要重复处理用户认证、表单验证、状态管理等基础逻辑。这些工作虽然不复杂,但每次从零开始确实会消耗不少时间。最近我发现用InsCode(快马)平台可以快速生成高质量的登录组件,大大提升了开发效率。 组件功能设…...
