9、组合模式(结构性模式)

        组合模式又叫部分整体模式,它创建了对象组的树形结构,将对象组合成树状结构,以一致的方式处理叶子对象以及组合对象,不以层次高低定义类,都是结点类

       一、传统组合模式

        举例,大学、学院、系,它们之间不是继承关系,是组合关系:大学由学院组成,学院由系组成,但是它们都是组织结点,一个大学,n个学院,一个学院m个系,大学、学院是 组合类型的,它们都包含结点成员,系是 叶子类型 的,不包含结点成员

        把大学类、学院类、系类抽象为组织结点类,它们都是组织结点类的子类(其中大学子类、学院子类是组合类型的类,系是叶子类型的类),按照具体依赖抽象原则,以子类共有的属性和行为来定义结点类,代码如下:

//组织结点(基类)Organization
class Organization{                      //具体依赖抽象原则
	std::string name;//数据成员
public:
	Organization( std::string name):name(name) { }//构造函数
	virtual ~Organization(){ }                    //虚析构
	//其他成员函数
    virtual std::string getName(){ return name; }   
    virtual void add(Organization* o){ throw "叶子结点没有成员!\n"; }
    //virtual void remove( ){ throw "叶子结点没有成员!\n";  }
    virtual void print() = 0;  //
};    

        组合类型的子类University、College,重写基类成员函数,代码几乎一样

//组合类型(子类)University College
//University
class University:public Organization{
	std::list<Organization*> ul; //以基类的形式包含其他子类对象
public:
	University(std::string name):Organization(name) { }
	~University(){
		std:: cout << "~University\n";
		for(auto o:ul){ delete o; }		
	}
	//重写
	virtual void add(Organization* o){
		ul.push_back(o);
	}
    virtual void print(){
		std::cout << getName() << "\n";//基类成员获取,基类的成员也是子类的组成部分
		for( auto o:ul ){
			o->print();
		}
	}	
};

//College
class College:public Organization{
	std::list<Organization*> cl; //以基类的形式包含其他子类对象
public:
	College(std::string name):Organization(name) { }
	~College(){
		std:: cout << "~College\n";
		for(auto o:cl){ delete o; }		
	}
	//重写
	virtual void add(Organization* o){
		cl.push_back(o);
	}
    virtual void print(){
		std::cout << getName() << "\n";//基类成员获取,基类的成员也是子类的组成部分
		for( auto o:cl ){
			o->print();
		}
	}	
};

        叶子类型的子类(Department)

//叶子类型(子类)Department
class Department:public Organization{
public:
	Department(std::string name):Organization(name) { }
	~Department(){
		std:: cout << "~Department\n";
	}
	//重写
	virtual void print(){
		std::cout << getName() << "\n";
	}
};

        头文件及测试代码

#include <iostream>
#include <string>
#include <exception>
#include <list>

int main() 
{
	University u("清华大学");
	Organization* c1 = new College("计算机学院");
	Organization* c2 = new College("信息工程学院");
	Organization* d = new Department("信息工程");
	u.add( c1 );
	u.add( c2 );
	c2->add( d );
	u.print();
	//c2->print();	
    return 0;
}

        二、组合模式的结点设计

        把组合类型的的属性、行为都放到结点,以对象组是否为空来判断是叶子类型的还是组合类型的,这样叶子类型也可以扩展为组合类型,虽然看起来有点混乱,但是确实可行,比如系原来是叶子,现在有了个学生会,添加成员后就可以不是叶子了

#include <iostream>
#include <string>
#include <exception>
#include <list>
 
//组织结点(基类)Organization
class Organization{  //具体依赖抽象原则
	std::string name;//数据成员
	std::list<Organization*> ol; //对象组
public:
	Organization( std::string name):name(name) { } //构造函数
	virtual ~Organization(){	 //虚析构
		for(auto o:ol){ delete o; }	
	}                    
	//其他成员函数 
    virtual void add(Organization* o){ ol.push_back(o); }	
    virtual void print(){
		std::cout << name << "\n";
		if( !ol.empty() ) //判断是否是叶子结点
		{ 
			for( auto o:ol ){
				o->print();
			}
		}	
	}
};    
 
//University
class University:public Organization{
public:
	University(std::string name):Organization(name) { }
	~University(){ std:: cout << "~University\n"; }
};
 
//College
class College:public Organization{
	std::list<Organization*> cl; //以基类的形式包含其他子类对象
public:
	College(std::string name):Organization(name) { }
	~College(){
		std:: cout << "~College\n";
	}
};
 
//Department
class Department:public Organization{
public:
	Department(std::string name):Organization(name) { }
	~Department(){
		std:: cout << "~Department\n";
	}
};
 
//StudentUnion
class StudentUnion:public Organization{
public:
	StudentUnion(std::string name):Organization(name) { }
	~StudentUnion(){
		std:: cout << "~StudentUnion\n";
	}
};
 
int main() 
{
	University u("清华大学");
	Organization* c1 = new College("计算机学院");
	Organization* c2 = new College("信息工程学院");
	Organization* d1 = new Department("信息工程");
	Organization* d2 = new Department("软件工程");
	Organization* s = new StudentUnion("信息工程学生会");
	
	d1->add( s );
	c1->add( d2 );
	c2->add( d1 );
	
	u.add( c1 );
	u.add( c2 );
	
	u.print( );
	//c2->print( );	
    return 0;
}

        三、修改打印函数,打印出树形结构

         1、 修改print

    virtual void print( int depth ){
		for(int i=0; i< depth; ++i)
			std:: cout << "--";
		std::cout << name << "\n";
		if( !ol.empty() ) //判断是否是叶子结点
		{ 
			for( auto o:ol ){
				o->print(depth+1);
			}
		}	
	}

        2、调用修改为

u.print(0);
//c2->print( 0 );

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/447589.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

崇法致行法律知识竞赛活动方案

赛程安排分两天&#xff0c;两场进行。 第一天&#xff08;第一场&#xff09;&#xff08;初赛&#xff09; 共 16 个二级分行&#xff0c;每行三人&#xff0c;共16 个战队参赛。 第一轮——必答轮 在大屏幕上显示10个选择题&#xff08;5个单选、5个多选&#xff09;&…

docker安装ollama

拉取镜像 docker pull ollama/ollama 运行容器 &#xff08;挂载路径 D:\ollama 改成你自己喜欢的路径&#xff09; CPU only docker run -d -v D:\ollama:/root/.ollama -p 11434:11434 --name ollama ollama/ollama Nvidia GPU&#xff08;没试过这个&#xff09; doc…

Vue脚手架

Vue脚手架 学习目标&#xff1a; 理解Node.js基本使用方法理解包资源管理器NPM的使用理解webpack的作用理解 vue-cli 脚手架 (重点)Element-UI 组件库 1.vue的格式&#xff1a;new Vue({//作用的视图el:"id选择器",//vue中的数据/*data:{key:value,key:value,...}…

判断链表回文

题目&#xff1a; //方法一&#xff0c;空间复杂度O(n) class Solution { public:bool isPalindrome(ListNode* head) {vector<int> nums; //放进数组后用双指针判断ListNode* cur head;while(cur){nums.emplace_back(cur->val);cur cur->next;}for(int i0…

Microsoft SQL Server 编写汉字转拼音函数

目录 应用场景 举例 函数实现 小结 应用场景 在搜索应用中&#xff0c;我们一般会提供一个搜索框&#xff0c;输入关健字&#xff0c;点击查询按钮以获取结果数据。大部分情况我们会提供模糊查询的形式以在一个或多个字段进行搜索以获取结果。这样可以简化用户的操作&…

高分1、2号卫星原始遥感影像数据

高分一号 高分一号卫高分一号卫星是中国高分辨率对地观测系统的首发星&#xff0c;突破了高空间分辨率、多光谱与宽覆盖相结合的光学遥感等关键技术&#xff0c;设计寿命5至8年。 高分辨率对地观测系统工程是《国家中长期科学和技术发展规划纲要(2006&#xff5e;2020年)》确定…

专题二 - 滑动窗口 - leetcode 1004. 最大连续1的个数 III | 中等难度

leetcode 1004. 最大连续1的个数 III leetcode 1004. 最大连续1的个数 III | 中等难度1. 题目详情1. 原题链接2. 基础框架 2. 解题思路1. 题目分析2. 算法原理3. 时间复杂度 3. 代码实现4. 知识与收获 leetcode 1004. 最大连续1的个数 III | 中等难度 1. 题目详情 给定一个二…

Linux之线程控制

目录 一、POSIX线程库 二、线程的创建 三、线程等待 四、线程终止 五、分离线程 六、线程ID&#xff1a;pthread_t 1、获取线程ID 2、pthread_t 七、线程局部存储&#xff1a;__thread 一、POSIX线程库 由于Linux下的线程并没有独立特有的结构&#xff0c;所以Linux并…

蓝牙系列十一:HCI层的数据格式

HCI层作为Host和Controlor链接的接口存在。以下是对HCI层的数据格式的解析。 1、参考&#xff1a;蓝牙协议core_v5.0.pdf 《Vol 2: Core System Package [BR/EDR Controller volume]》的“Part E: Host Controller Interface Functional Specification” 2. 协议栈框图 对于被…

Linux:kubernetes(k8s)pod的基础操作(6)

Linux&#xff1a;kubernetes&#xff08;k8s&#xff09;允许在任意节点使用kubectl命令&#xff08;5&#xff09;-CSDN博客https://blog.csdn.net/w14768855/article/details/136460090?spm1001.2014.3001.5501 我在前两张进行了基础环境的一系列搭建&#xff0c;现在就正…

NIFI从Oracle11G同步数据到Mysql_亲测可用_解决数据重复_数据跟源表不一致的问题---大数据之Nifi工作笔记0065

首先来看一下整体的流程: 可以看到了用到了上面的这些处理器,然后我们主要看看,这里之前 同步的时候,总是出现重复的数据,奇怪. 比如源表中只有166条数据,但是同步过去以后变成了11万条数据了. ${db.table.name:equals(table1):or(${db.table.name:equals(table2)})} 可以看…

职场成功的关键:积极主动,勇于担当

在职场中&#xff0c;每个人都渴望成功。然而&#xff0c;成功并非一蹴而就&#xff0c;而是需要我们在日常工作中不断积累、锻炼和提升。本文将为您揭示职场成功的关键因素&#xff0c;帮助您在职场道路上越走越远。 一、积极主动&#xff0c;主动承担责任 在职场中&#xff0…

吴恩达机器学习笔记十六 如何debug一个学习算法 模型评估 模型选择和训练 交叉验证测试集

如果算法预测出的结果不太好&#xff0c;可以考虑以下几个方面&#xff1a; 获得更多的训练样本 采用更少的特征 尝试获取更多的特征 增加多项式特征 增大或减小 λ 模型评估(evaluate model) 例如房价预测&#xff0c;用五个数据训练出的模型能很好的拟合这几个数据&am…

RocketMQ入门指南:从零开始学习分布式消息队列技术

RocketMQ 1. MQ介绍1.1 为什么要用MQ1.2 MQ的优点和缺点1.3 各种MQ产品的比较 2. RocketMQ快速入门2.1 准备工作2.1.1 下载RocketMQ2.2.2 环境要求 2.2 安装RocketMQ2.2.1 安装步骤2.2.2 目录介绍 2.3 启动RocketMQ2.4 测试RocketMQ2.4.1 发送消息2.4.2 接收消息 2.5 关闭Rocke…

【Python】科研代码学习:七 TrainingArguments,Trainer

【Python】科研代码学习&#xff1a;七 TrainingArguments&#xff0c;Trainer TrainingArguments重要的方法 Trainer重要的方法使用 Trainer 的简单例子 TrainingArguments HF官网API&#xff1a;Training 众所周知&#xff0c;推理是一个大头&#xff0c;训练是另一个大头 之…

【PyTorch实战演练】深入剖析MTCNN(多任务级联卷积神经网络)并使用30行代码实现人脸识别

文章目录 0. 前言1. 级联神经网络介绍2. MTCNN介绍2.1 MTCNN提出背景2.2 MTCNN结构 3. MTCNN PyTorch实战3.1 facenet_pytorch库中的MTCNN3.2 识别图像数据3.3 人脸识别3.4 关键点定位 0. 前言 按照国际惯例&#xff0c;首先声明&#xff1a;本文只是我自己学习的理解&#xff…

【Qt学习笔记】(二)--第一个程序“Hello World”(学习Qt中程序的运行、发布、编译过程)

声明&#xff1a;本人水平有限&#xff0c;博客可能存在部分错误的地方&#xff0c;请广大读者谅解并向本人反馈错误。    因为我个人对Qt也是有一些需求&#xff0c;所以开设本专栏进行学习&#xff0c;希望大家可以一起学习&#xff0c;共同进步。   这篇博客将从一个 He…

算法刷题Day1 | 704.二分查找、27.移除元素

目录 0 引言1 二分查找1.1 我的解题1.2 修改后1.3 总结 2 移除元素2.1 暴力求解2.2 双指针法&#xff08;快慢指针&#xff09; &#x1f64b;‍♂️ 作者&#xff1a;海码007&#x1f4dc; 专栏&#xff1a;算法专栏&#x1f4a5; 标题&#xff1a;代码随想录算法训练营第一天…

Vue.js+SpringBoot开发大病保险管理系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 系统配置维护2.2 系统参保管理2.3 大病保险管理2.4 大病登记管理2.5 保险审核管理 三、系统详细设计3.1 系统整体配置功能设计3.2 大病人员模块设计3.3 大病保险模块设计3.4 大病登记模块设计3.5 保险审核模块设计 四、…

MySQL三种日志

一、undo log&#xff08;回滚日志&#xff09; 1.作用&#xff1a; &#xff08;1&#xff09;保证了事物的原子性 &#xff08;2&#xff09;通过read view和undo log实现mvcc多版本并发控制 2.在事务提交前&#xff0c;记录更新前的数据到undo log里&#xff0c;回滚的时候读…