php设计模式-组合模式的运用

介绍
PHP的组合模式是一种设计模式,用于将对象组合成树形结构以表示“部分-整体”的层次结构。该模式允许客户端统一处理单个对象和组合对象,使得客户端在处理对象时不需要知道对象是否为单个对象还是组合对象。
在组合模式中,有两种类型的对象:组合对象和叶子对象。组合对象是由多个叶子对象组成的对象,它具有与叶子对象相同的接口。而叶子对象是组合对象中最基本的对象,它不能再包含其他对象。
组合模式通常由以下几个角色组成:
抽象组件(Component):定义组合中所有对象的通用接口,可以是抽象类或接口。
叶子组件(Leaf):表示组合中的叶子节点对象,它不能包含任何子对象。
组合组件(Composite):表示组合中的非叶子节点对象,它通常包含多个叶子节点或其他组合对象。
客户端(Client):通过抽象组件接口操作组合中的对象,而不需要知道具体的对象类型。
使用组合模式可以方便地处理层次结构对象,使得客户端可以透明地处理单个对象和组合对象,从而简化了客户端的代码实现。例如,在处理一个树形结构的目录时,可以使用组合模式来方便地处理整个目录结构,无需关注目录中的每个文件或子目录。
场景
组合模式适用于以下场景:
树形结构:当我们需要处理具有层次结构的对象时,可以使用组合模式。例如,图形用户界面中的窗口、控件和组合控件就可以使用组合模式来实现。
部分-整体关系:当我们需要将一些对象视为一个整体时,并且希望能够像处理单个对象一样处理整体,可以使用组合模式。例如,汽车零件和汽车可以被组合成一个整体,并且可以像处理单个零件一样处理整个汽车。
统一接口:当我们需要将对象和组合对象视为具有统一接口的对象时,可以使用组合模式。例如,在文件系统中,文件和目录都有相同的操作接口,因此可以使用组合模式来实现它们。
需要忽略对象集合与单个对象之间的差异:当我们需要忽略对象集合与单个对象之间的差异时,可以使用组合模式。例如,将公司中的所有员工视为相同的对象,以便能够像处理单个员工一样处理整个公司。
可以动态添加和删除对象:当我们需要能够动态添加和删除对象时,可以使用组合模式。例如,在菜单系统中,菜单和子菜单可以被动态添加和删除,因此可以使用组合模式来实现它们。
运用

组合模式在购物车的运用
在购物车中,我们需要处理一组商品对象,这些商品对象可以是单个商品或商品的组合,例如套餐或礼篮。同时,我们需要能够对购物车中的商品进行添加、删除和计算总价等操作,这些操作可以使用组合模式来实现。
我们可以定义一个抽象类或接口,表示商品和商品组合的公共行为,例如计算价格和获取商品列表。然后,我们可以创建具体的商品类和商品组合类,它们都实现了这个抽象类或接口。商品组合类中包含一组商品对象,可以通过添加或删除单个商品或商品组合来构建更复杂的商品组合。这种嵌套结构形成了一棵树形结构,树的节点是商品或商品组合。
最后,我们可以创建一个购物车类,它包含购物车中的所有商品对象,并提供添加、删除和计算总价等操作。购物车类中的操作可以递归地遍历商品组合树,处理每个商品对象。这样,我们就可以使用组合模式来实现购物车逻辑。
// 抽象类或接口
abstract class Product {abstract public function getPrice();abstract public function getProducts();
}// 商品类
class Item extends Product {private $name;private $price;public function __construct($name, $price) {$this->name = $name;$this->price = $price;}public function getPrice() {return $this->price;}public function getProducts() {return array($this);}
}// 商品组合类
class Combo extends Product {private $name;private $products = array();public function __construct($name) {$this->name = $name;}public function addProduct(Product $product) {$this->products[] = $product;}public function removeProduct(Product $product) {$key = array_search($product, $this->products, true);if ($key !== false) {unset($this->products[$key]);}}public function getPrice() {$totalPrice = 0;foreach ($this->products as $product) {$totalPrice += $product->getPrice();}return $totalPrice;}public function getProducts() {return $this->products;}
}// 购物车类
class ShoppingCart {private $products = array();public function addProduct(Product $product) {$this->products[] = $product;}public function removeProduct(Product $product) {$key = array_search($product, $this->products, true);if ($key !== false) {unset($this->products[$key]);}}public function getTotalPrice() {$totalPrice = 0;foreach ($this->products as $product) {$totalPrice += $product->getPrice();}return $totalPrice;}
}// 示例用法
$item1 = new Item('Product A', 10);
$item2 = new Item('Product B', 20);
$item3 = new Item('Product C', 30);$combo1 = new Combo('Combo 1');
$combo1->addProduct($item1);
$combo1->addProduct($item2);$combo2 = new Combo('Combo 2');
$combo2->addProduct($item2);
$combo2->addProduct($item3);$cart = new ShoppingCart();
$cart->addProduct($item1);
$cart->addProduct($combo1);
$cart->addProduct($combo2);$totalPrice = $cart->getTotalPrice();
echo "Total Price: $totalPrice";
说明
在上面的购物车代码中,Product 抽象类或接口表示商品和商品组合的公共行为,Item 类表示单个商品,Combo 类表示商品组合,ShoppingCart 类表示购物车。因此,在这个示例中,Item 和 Combo 是叶子组件,Combo 是容器组件。
具体来说,Item 是叶子组件,因为它是商品组合树的最底层,没有子组件。它只包含单个商品,提供了获取商品价格和获取商品列表的方法。
而 Combo 是容器组件,它包含了一组商品对象,包括单个商品和其他商品组合。它提供了添加、删除商品和计算价格等方法。因此,它既可以是容器组件,也可以是叶子组件,具体取决于它是否包含其他商品或商品组合。
在购物车中,ShoppingCart 类是容器组件,它包含了购物车中的所有商品对象,包括单个商品和商品组合。它提供了添加、删除商品和计算总价等方法。因此,ShoppingCart 类既可以是容器组件,也可以是叶子组件,具体取决于它是否包含其他商品或商品组合。
综上,根据组合模式的定义,叶子组件表示树形结构中最底层的对象,容器组件表示树形结构中非最底层的对象,它们都实现了相同的接口或抽象类,从而使客户端可以一致地对待它们。在购物车中,商品对象和商品组合对象都实现了 Product 接口,因此它们可以在 ShoppingCart 中互相嵌套,从而形成了一棵树形结构。
相关文章:
php设计模式-组合模式的运用
介绍 PHP的组合模式是一种设计模式,用于将对象组合成树形结构以表示“部分-整体”的层次结构。该模式允许客户端统一处理单个对象和组合对象,使得客户端在处理对象时不需要知道对象是否为单个对象还是组合对象。 在组合模式中,有两种类型的…...
一文教会你如何简单使用Fegin进行远程服务调用
文章目录1、fegin的基本介绍2、fegin的基本使用步骤3、项目中的实际运用4、测试前言在分布式微服务中,少不了会进行不同服务之间的相互调用,比如A服务要调用B服务中的接口,如何简单方便的实现呢?fegin可以来帮助。 1、fegin的基本…...
OpenAI——CLIPs(代码使用示例)
OpenAI——CLIPs(打通NLP与CV) Open AI在2021年1月份发布Contrastive Language-Image Pre-training(CLIP),基于对比文本-图像对对比学习的多模态模型,通过图像和它对应的文本描述对比学习,模型能够学习到文本-图像对的匹配关系。它开源、多模态、zero-s…...
什么样的人更适合创业?那类人创业更容易成功?
创业是一项充满风险和机遇的事业,成功的创业者需要具备一定的素质和能力。那么,什么样的人更适合创业?哪类人创业更容易成功呢?本文将为您介绍几个适合创业的人群和成功创业者的共同特点。 具有创新精神的人 创业需要不断创新&am…...
JavaApi操作ElasticSearch(强烈推荐)
ElasticSearch 高级 1 javaApi操作es环境搭建 在elasticsearch官网中提供了各种语言的客户端:https://www.elastic.co/guide/en/elasticsearch/client/index.html 而Java的客户端就有两个: 不过Java API这个客户端(Transport Client&#…...
NFT的前景,元宇宙的发展
互联网的普及和数字技术的广泛应用,成为消费升级的新动力,在不断创造出更好的数字化生活的同时,也改变了人们的消费习惯、消费内容、消费模式,甚至是消费理念,数字经济时代的文化消费呈现出新的特征。 2020年有关机构工…...
C#基础教程20 预处理器指令
文章目录 C#预处理指令教程简介预处理指令格式指令名 参数预处理指令类型条件编译指令if#if 条件表达式宏定义指令总结C#预处理指令教程 简介 预处理指令是在编译代码之前进行的一种处理,可以让程序员在编译前根据需要对代码进行一些修改、调整或者控制。C#语言中的预处理指令…...
【FPGA】Verilog:时序电路设计 | 二进制计数器 | 计数器 | 分频器 | 时序约束
前言:本章内容主要是演示Vivado下利用Verilog语言进行电路设计、仿真、综合和下载 示例:计数器与分频器 功能特性: 采用 Xilinx Artix-7 XC7A35T芯片 配置方式:USB-JTAG/SPI Flash 高达100MHz 的内部时钟速度 存储器&#…...
国外SEO策略指南:确保你的网站排名第一!
如果你想在谷歌等搜索引擎中获得更高的排名并吸引更多的流量和潜在客户,那么你需要了解一些国外SEO策略。 下面是一些可以帮助你提高谷歌排名的关键策略。 网站结构和内容优化 谷歌的搜索算法会考虑网站的结构和内容。 因此,你需要优化网站结构&…...
Tik Tok新手秘籍,做好五点可轻松起号
新手做TikTok需要有一个具体的规划布局,如果没有深思熟虑就上手开始的话,很有可能会导致功亏一篑,甚至是浪费时间。因此,想要做好 TikTok,就必须从最基本的运营细节开始,一步一步来,下面为大家分…...
【Linux】网络入门
🎇Linux: 博客主页:一起去看日落吗分享博主的在Linux中学习到的知识和遇到的问题博主的能力有限,出现错误希望大家不吝赐教分享给大家一句我很喜欢的话: 看似不起波澜的日复一日,一定会在某一天让你看见坚持…...
回溯法——力扣题型全解【更新中】
(本文源自网上教程的笔记) 回溯基础理论 回溯搜索法,它是一种搜索的方式。 回溯是递归的副产品,只要有递归就会有回溯。 所以以下讲解中,回溯函数也就是递归函数,指的都是一个函数。 回溯法的效率 虽然…...
【华为机试真题详解 Python实现】分奖金【2023 Q1 | 100分】
文章目录 前言题目描述输入描述输出描述示例 1题目解析参考代码前言 《华为机试真题详解》专栏含牛客网华为专栏、华为面经试题、华为OD机试真题。 如果您在准备华为的面试,期间有想了解的可以私信我,我会尽可能帮您解答,也可以给您一些建议! 本文解法非最优解(即非性能…...
netlink进行网卡重命名
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <errno.h> #include <unistd.h> #include <sys/socket.h> #include <linux/if.h> #include <linux/netlink.h>#define MAX_PAYLOAD 1024 // 最大负载长…...
2023年春【数据分析与挖掘】文献精读(一)-1:针对COVID-19,使用聚类方法有效提取生物特性关联进而识别预防COVID-19的药物
分享给大家——动漫《画江湖之不良人》第四季片尾,主人公 李星云所说的一段话: 悠悠众生,因果循环,大道至简,世间若尽是不如意事, 越是执着,便越是苦,不如安下心来,看该看的风景,做好该做之事。 初行娆疆,所悟如此, 就像曾经有一位紫衣姑娘,第一次来中原时,一样…...
【Go自学第三节】Go的范围(Range)用法
Go 语言中 range 关键字用于 for 循环中迭代数组(array)、切片(slice)、通道(channel)或集合(map)的元素。在数组和切片中它返回元素的索引和索引对应的值,在集合中返回 key-value 对。 在讲Go语言的range之前,我们先回顾下Python中range的用法 for i …...
【备战面试】每日10道面试题打卡-Day6
本篇总结的是计算机网络知识相关的面试题,后续也会更新其他相关内容 文章目录1、HTTP 与 HTTPS 有哪些区别?2、HTTPS的加密过程是什么?3、GET与POST有什么区别?4、讲讲HTTP各个版本的区别?5、HTTP与FTP的区别ÿ…...
Stable Diffusion 个人推荐的各种模型及设置参数、扩展应用等合集(不断更新中)
一、说明 | 表示或者 表示 以上 二、模型 适用风景、房子、车子等漫画类风格 模型的VAE不要用模型附带的,好像就是naifu的官方vae,很老了,用 vae-ft-mse-840000-ema-pruned.ckpt 或者是 kl-f8-anime2.ckpt; 嵌入模型要下载作者…...
Salesforce 2023财年逆风增长,现金流达历史最高!
在过去的一年里,Salesforce一直是华尔街最关注的公司之一。3月1日,CRM领域的全球领导者Salesforce公布了截至2023年1月31日的第四季度和整个财年的业绩。 Salesforce主席兼首席执行官Marc Benioff表示: Salesforce全年实现了314亿美元的收入…...
2023年3月全国数据治理工程师认证DAMA-CDGA/CDGP考试怎么通过?
弘博创新是DAMA中国授权的数据治理人才培养基地,贴合市场需求定制教学体系,采用行业资深名师授课,理论与实践案例相结合,快速全面提升个人/企业数据治理专业知识与实践经验,通过考试还能获得数据专业领域证书。 DAMA认…...
使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式
一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明:假设每台服务器已…...
树莓派超全系列教程文档--(62)使用rpicam-app通过网络流式传输视频
使用rpicam-app通过网络流式传输视频 使用 rpicam-app 通过网络流式传输视频UDPTCPRTSPlibavGStreamerRTPlibcamerasrc GStreamer 元素 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 使用 rpicam-app 通过网络流式传输视频 本节介绍来自 rpica…...
以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:
一、属性动画概述NETX 作用:实现组件通用属性的渐变过渡效果,提升用户体验。支持属性:width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项: 布局类属性(如宽高)变化时&#…...
Java求职者面试指南:Spring、Spring Boot、MyBatis框架与计算机基础问题解析
Java求职者面试指南:Spring、Spring Boot、MyBatis框架与计算机基础问题解析 一、第一轮提问(基础概念问题) 1. 请解释Spring框架的核心容器是什么?它在Spring中起到什么作用? Spring框架的核心容器是IoC容器&#…...
使用Spring AI和MCP协议构建图片搜索服务
目录 使用Spring AI和MCP协议构建图片搜索服务 引言 技术栈概览 项目架构设计 架构图 服务端开发 1. 创建Spring Boot项目 2. 实现图片搜索工具 3. 配置传输模式 Stdio模式(本地调用) SSE模式(远程调用) 4. 注册工具提…...
排序算法总结(C++)
目录 一、稳定性二、排序算法选择、冒泡、插入排序归并排序随机快速排序堆排序基数排序计数排序 三、总结 一、稳定性 排序算法的稳定性是指:同样大小的样本 **(同样大小的数据)**在排序之后不会改变原始的相对次序。 稳定性对基础类型对象…...
FFmpeg:Windows系统小白安装及其使用
一、安装 1.访问官网 Download FFmpeg 2.点击版本目录 3.选择版本点击安装 注意这里选择的是【release buids】,注意左上角标题 例如我安装在目录 F:\FFmpeg 4.解压 5.添加环境变量 把你解压后的bin目录(即exe所在文件夹)加入系统变量…...
【题解-洛谷】P10480 可达性统计
题目:P10480 可达性统计 题目描述 给定一张 N N N 个点 M M M 条边的有向无环图,分别统计从每个点出发能够到达的点的数量。 输入格式 第一行两个整数 N , M N,M N,M,接下来 M M M 行每行两个整数 x , y x,y x,y,表示从 …...
C/Python/Go示例 | Socket Programing与RPC
Socket Programming介绍 Computer networking这个领域围绕着两台电脑或者同一台电脑内的不同进程之间的数据传输和信息交流,会涉及到许多有意思的话题,诸如怎么确保对方能收到信息,怎么应对数据丢失、被污染或者顺序混乱,怎么提高…...
记一次spark在docker本地启动报错
1,背景 在docker中部署spark服务和调用spark服务的微服务,微服务之间通过fegin调用 2,问题,docker容器中服务器来后,注册中心都有,调用服务也正常,但是调用spark启动任务后报错,报错…...
