设计模式:原型模式(C#、JAVA、JavaScript、C++、Python、Go、PHP)

上一篇《访问者模式》                                                                   下一篇《享元模式》

简介:

原型模式,它是一种创建型设计模式,它允许通过复制原型对象来创建新的对象,而无需知道创建的细节。其工作原理是将一个原型对象传递给要创建的对象,然后通过请求原型对象复制自己来实施创建。

在原型模式中,克隆方法所创建的对象是全新对象,它们在内存中拥有全新的地址,通常对克隆所产生的对象进行修改不会对原型对象造成任何影响,每个克隆对象都是相互独立的。通过不同的方式对克隆对象进行修改后,可以得到一系列相似但不完全相同的对象。

需要注意的是,对原型对象的浅拷贝,对于数据类型是基本数据类型的成员变量,会直接进行值传递,也就是将该属性值复制一份给新的对象;对于数据类型是引用数据类型的成员变量,比如说成员变量是某个数组、某个类的对象等,那么浅拷贝会进行引用传递,也就是只是将该成员变量的引用值(内存地址)复制一份给新的对象。在这种情况下,在一个对象中修改该成员变量会影响到另一个对象的该成员变量值。

原型模式的使用场景:
1、类初始化需要消化非常多的资源,包括数据、硬件资源等。通过原型拷贝可以避免这些消耗。
2、通过使用new关键字创建一个对象需要非常繁琐的数据准备或访问权限。
3、一个对象需要提供给其他对象访问,而且各个调用者可能需要修改其值。在这种情况下,可以考虑使用原型模式拷贝多个对象供调用者使用,即保护性拷贝。

在实际项目中,原型模式很少单独出现,一般是和工厂方法模式一起出现,通过 clone 的方法创建一个对象,然后由工厂方法提供给调用者。

原型模式的创建步骤:
1、定义抽象原型类:抽象原型类是定义具有克隆自己的方法的接口,是所有具体原型类的公共父类。
2、定义具体原型类:具体原型类实现抽象原型类中的克隆方法,返回自己的一个克隆对象。
3、定义客户类:客户类让一个原型克隆自身,从而创建一个新的对象。在客户类中只需要直接实例化或通过工厂方法等创建一个对象,再通过调用该对象的克隆方法复制得到多个相同的对象。

原型模式通过复制原型对象来创建新对象,减少了创建新对象时所消耗的时间和资源。同时,由于复制的是原型对象,因此不会影响原对象的状态。

原型模式的优点,主要包括:
1、简化创建过程:原型模式可以简化对象的创建过程,通过复制一个已有实例可以提高新实例的创建效率。
2、扩展性较好:由于在原型模式中提供了抽象原型类,在客户端可以针对抽象原型类进行编程,而将具体原型类写在配置文件中,增加或减少产品类对原有系统都没有任何影响。
3、提供简化的创建结构:原型模式中产品的复制是通过封装在原型类中的克隆方法实现的,无须专门的工厂类来创建产品。
4、支持深拷贝:原型模式可以使用深克隆方式保存对象的状态,使用原型模式将对象复制一份,并将其状态保存起来,简化了创建对象的过程,以便在需要的时候使用(如恢复到某一历史状态),可辅助实现撤销操作。

总的来说,原型模式可以大大提高创建对象的效率,同时还能保证系统的扩展性和灵活性。

原型模式的缺点,主要包括:
1、在实现深拷贝时可能需要比较复杂的代码,需要为每一个类配备一个克隆方法,而且该克隆方法需要对类的功能进行通盘考虑,这对全新的类来说不是很难,但对已有的类进行改造时,不一定是件容易的事,必须修改其源代码,违背了“开闭原则”。

示例:

一、C#原型模式

以下是一个示例,展示了如何在C#中实现原型模式:

using System;  namespace PrototypePatternExample  
{  // 抽象原型类  public abstract class Prototype  {  public abstract Prototype Clone();  }  // 具体原型类  public class ConcretePrototype : Prototype  {  private string _property;  public ConcretePrototype(string property)  {  _property = property;  }  public override Prototype Clone()  {  // 使用深拷贝复制对象  ConcretePrototype clone = (ConcretePrototype)MemberwiseClone(this);  return clone;  }  public void Display()  {  Console.WriteLine("Property: " + _property);  }  }  class Program  {  static void Main(string[] args)  {  // 创建原型对象  ConcretePrototype prototype = new ConcretePrototype("Hello World");  // 克隆原型对象  ConcretePrototype clonePrototype = (ConcretePrototype)prototype.Clone();  // 显示原型对象和克隆对象的属性值是否相同  prototype.Display();  clonePrototype.Display();  Console.ReadLine();  }  }  
}

二、java原型模式

原型模式通常通过以下方式实现:

import java.util.ArrayList;  
import java.util.List;    abstract class Shape {  private String name;  public Shape(String name) {  this.name = name;  }  public String getName() {  return name;  }  public abstract void draw();  // 实现克隆方法  public Shape clone() {  try {  return (Shape) this.getClass().newInstance();  } catch (Exception e) {  e.printStackTrace();  return null;  }  }  
}  class Circle extends Shape {  private int radius;  public Circle(String name, int radius) {  super(name);  this.radius = radius;  }  @Override  public void draw() {  System.out.println("Drawing Circle");  }  
}  class Rectangle extends Shape {  private int width;  private int height;  public Rectangle(String name, int width, int height) {  super(name);  this.width = width;  this.height = height;  }  @Override  public void draw() {  System.out.println("Drawing Rectangle");  }  
}public class PrototypePatternDemo {  public static void main(String[] args) {  List<Shape> shapeList = new ArrayList<>();  shapeList.add(new Circle("Circle 1", 5));  shapeList.add(new Rectangle("Rectangle 1", 5, 10));  shapeList.add(new Circle("Circle 2", 10));  shapeList.add(new Rectangle("Rectangle 2", 10, 20));  // 使用原型模式创建对象,节省创建对象的时间  for (Shape shape : shapeList) {  Shape cloneShape = shape.clone();  System.out.println("Original Shape: " + shape.getName());  System.out.println("Clone Shape: " + cloneShape.getName());  }  }  
}  

三、javascript原型模式

在JavaScript中,实现原型模式的关键是使用构造函数和原型对象。
下面是一个简单的JavaScript原型模式示例:

// 定义原型对象  
var CarProto = {  color: "blue",  speed: 0,  start: function() {  console.log("Car started");  },  stop: function() {  console.log("Car stopped");  }  
};  // 定义构造函数  
function Car(color) {  this.color = color;  this.speed = 0;  // 将构造函数prototype属性指向原型对象  this.prototype = CarProto;  
}  // 定义子类  
function SportCar() {  // 调用父类构造函数  Car.call(this, "red");  // 重写父类方法  this.start = function() {  console.log("SportCar started");  };  
}  // 设置原型对象,让SportCar继承CarProto  
SportCar.prototype = CarProto;

在这个示例中,我们定义了一个CarProto原型对象,它包含了汽车的属性和方法。然后我们定义了一个Car构造函数,它接受颜色参数,并设置速度属性为0,并将它的prototype属性指向CarProto。这样,当我们创建一个新的汽车对象时,它就会继承CarProto的属性和方法。

然后我们定义了一个SportCar子类,它调用父类构造函数,并重写了父类的start方法。最后我们将SportCar.prototype设置为CarProto,这样SportCar就可以继承CarProto的属性和方法了。现在我们可以使用new关键字来创建SportCar对象了。

四、C++原型模式

以下是在C++中实现原型模式:

#include <iostream>  
#include <string>  
#include <map>  using namespace std;  // 定义原型接口  
class Prototype {  
public:  virtual Prototype* clone() = 0;  virtual void display() = 0;  
};  // 具体原型类  
class ConcretePrototype : public Prototype {  
private:  string name;  
public:  ConcretePrototype(string n) : name(n) {}  void setName(string n) { name = n; }  string getName() { return name; }  // 实现克隆方法  Prototype* clone() { return new ConcretePrototype(*this); }  // 实现显示方法  void display() { cout << "ConcretePrototype " << name << endl; }  
};  // 工厂类  
class PrototypeFactory {  
private:  map<string, Prototype*> prototypes; // 存储原型对象的映射表  
public:  PrototypeFactory() {}  Prototype* create(string type) { // 创建原型对象  if (prototypes.find(type) == prototypes.end()) { // 如果该类型的原型对象不存在,则创建并存储在映射表中  prototypes[type] = new ConcretePrototype(type);  }  return prototypes[type]->clone(); // 返回克隆后的对象  }  
};  int main() {  PrototypeFactory factory;  Prototype* p1 = factory.create("prototype1"); // 创建原型对象1的克隆对象1  p1->display(); // 显示原型对象1的名称,输出 "ConcretePrototype prototype1"  Prototype* p2 = factory.create("prototype1"); // 创建原型对象1的克隆对象2  p2->setName("prototype2"); // 设置克隆对象2的名称,不影响原型对象1的名称  p2->display(); // 显示原型对象1的名称,输出 "ConcretePrototype prototype1",因为克隆对象2的名称没有修改成功,仍然是原型对象1的名称  delete p1; // 释放内存空间,因为p1和p2都是通过克隆得到的,所以应该释放内存空间,避免内存泄漏问题。  delete p2; // 释放内存空间,因为p1和p2都是通过克隆得到的,所以应该释放内存空间,避免内存泄漏问题。  return 0;  
}

五、python原型模式

以下是在python中实现原型模式:

import copy  # 定义原型类  
class Prototype:  def __init__(self, name):  self.name = name  def clone(self):  return copy.deepcopy(self)  # 定义具体原型类  
class ConcretePrototype(Prototype):  def __init__(self, name):  super().__init__(name)  self.data = []  def add_data(self, data):  self.data.append(data)  def clone(self):  return ConcretePrototype(self.name)  # 测试代码  
if __name__ == '__main__':  # 创建原型对象  prototype1 = ConcretePrototype("prototype1")  prototype1.add_data(1)  prototype1.add_data(2)  print("Prototype 1 data:", prototype1.data)  # 克隆原型对象  clone1 = prototype1.clone()  clone1.add_data(3)  print("Clone 1 data:", clone1.data) # [1, 2, 3]  print("Prototype 1 data:", prototype1.data) # [1, 2]  # 克隆克隆对象,避免修改原对象的影响  clone2 = clone1.clone()  clone2.add_data(4)  print("Clone 2 data:", clone2.data) # [1, 2, 4]  print("Prototype 1 data:", prototype1.data) # [1, 2]

六、go原型模式

以下是一个示例,展示了如何在go中实现原型模式:

package main  import (  "fmt"  
)  // 原型接口  
type Prototype interface {  Clone() Prototype  
}  // 具体原型类  
type ConcretePrototype struct {  Name string  
}  // 克隆方法实现原型接口  
func (p *ConcretePrototype) Clone() Prototype {  return &ConcretePrototype{Name: p.Name}  
}  func main() {  // 创建原型对象  prototype1 := &ConcretePrototype{Name: "Prototype1"}  // 克隆原型对象  clone1 := prototype1.Clone()  fmt.Println("Clone 1 Name:", clone1.(*ConcretePrototype).Name) // 输出:Clone 1 Name: Prototype1  // 修改原型对象  prototype1.Name = "Prototype2"  fmt.Println("Prototype 1 Name:", prototype1.(*ConcretePrototype).Name) // 输出:Prototype 1 Name: Prototype2  // 克隆克隆对象,避免修改原对象的影响  clone2 := clone1.Clone()  fmt.Println("Clone 2 Name:", clone2.(*ConcretePrototype).Name) // 输出:Clone 2 Name: Prototype1  
}

七、PHP原型模式

以下是一个示例,展示了如何在PHP中实现原型模式:

<?php  class Prototype implements Cloneable {  private $name;  public function __construct($name) {  $this->name = $name;  }  public function getName() {  return $this->name;  }  public function setName($name) {  $this->name = $name;  }  public function clone() {  return clone $this;  }  
}  // 创建原型对象  
$prototype = new Prototype("Original");  
echo "Prototype Name: " . $prototype->getName() . "\n";  // 克隆原型对象  
$clone = $prototype->clone();  
echo "Clone Name: " . $clone->getName() . "\n";  // 修改原型对象的属性  
$prototype->setName("Modified");  
echo "Prototype Name after modification: " . $prototype->getName() . "\n";  // 克隆克隆对象,避免修改原对象的影响  
$clone2 = $clone->clone();  
echo "Clone 2 Name: " . $clone2->getName() . "\n";


《完结》

上一篇《访问者模式》                                                                     下一篇《享元模式》

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

当前JavaEE初阶的阶段知识总结

当前JavaEE初阶的阶段知识总结 多线程 文件IO 文件系统操作 ~~ File类. 文件内容操作 ~~ 读文件,写文件. IO 流对象. 流(Stream),形象的比喻,读取文件,就像水流一样,读写文件的时候,和水流类似,读100字节,可以一次读1个字节,100次完成;也可以一次读10个字节,10次完成…… 在…

基于Java的婚纱摄影网站系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09; 代码参考数据库参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者&am…

【数据结构】排序算法总结

⭐ 作者&#xff1a;小胡_不糊涂 &#x1f331; 作者主页&#xff1a;小胡_不糊涂的个人主页 &#x1f4c0; 收录专栏&#xff1a;浅谈数据结构 &#x1f496; 持续更文&#xff0c;关注博主少走弯路&#xff0c;谢谢大家支持 &#x1f496; 总结 1. 归并排序2. 排序算法复杂度…

飞利浦双串口51单片机485网关

主要功能将PC端的数据接收下来&#xff0c;分发到不同的设备&#xff0c;也是轮询设备数据读取回来&#xff0c;打包回传到PC端&#xff0c;数据包包头包尾识别&#xff0c;数据校验&#xff0c;接收超时处理&#xff0c;将协议结构化处理&#xff0c;协议的改动不需要改动程序…

Python 算法高级篇:图的表示与存储优化

Python 算法高级篇&#xff1a;图的表示与存储优化 引言 1. 什么是图&#xff1f;2. 图的基本概念3. 图的表示方法3.1. 临接矩阵表示临接矩阵的优点&#xff1a;临接矩阵的缺点&#xff1a; 3.2. 邻接表表示邻接表的优点&#xff1a;邻接表的缺点&#xff1a; 4. 优化的存储方法…

【开源】基于SpringBoot的计算机机房作业管理系统的设计和实现

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 登录注册模块2.2 课程管理模块2.3 课时管理模块2.4 学生作业模块 三、系统设计3.1 用例设计3.2 数据库设计3.2.1 课程表3.2.2 课时表3.2.3 学生作业表 四、系统展示五、核心代码5.1 查询课程数据5.2 新增课时5.3 提交作…

C++初阶--类与对象(2)

文章目录 类的6个默认成员函数构造函数默认构造函数的用处&#xff1a; 析构函数默认析构函数的用处 拷贝构造函数赋值运算符重载取地址及const取地址操作符重载 类的6个默认成员函数 对于一个没有成员的类来说&#xff0c;称之为空类&#xff1b; 但空类并非完全是空的&#…

1024 云上见 · 上云挑战(ChatGPT搭建)

【玩转1024】使用函数计算X通义千问搭建AI助手&#xff0c;参与1024小说创作大赛 【使用函数计算X通义千问搭建AI助手&#xff0c;参与小说创作大赛】&#xff1a;本活动基于函数计算X 通义千问快速部署 AI 个人助手应用&#xff0c;用户可以根据需要选择不同角色的AI助手开启…

Maven配置阿里云中央仓库settings.xml

Maven配置阿里云settings.xml 前言一、阿里云settings.xml二、使用步骤1.任意目录创建settings.xml2.使用阿里云仓库 总结 前言 国内网络从maven中央仓库下载文件通常是比较慢的&#xff0c;所以建议配置阿里云代理镜像以提高jar包下载速度&#xff0c;IDEA中我们需要配置自己…

基础课13——数据异常处理

数据异常是指数据不符合预期或不符合常识的情况。数据异常可能会导致数据分析结果不准确&#xff0c;甚至是错误&#xff0c;因此在进行数据分析之前需要对数据进行清洗和验证。 常见的数据异常包括缺失值、重复值、异常值等。 缺失值是指数据中存在未知值或未定义的值&#…

JAVA:集合框架常见的面试题和答案

1、List接口的常见实现类有哪些&#xff1f; 答&#xff1a; 常见的List接口实现类包括&#xff1a; ArrayList: 基于动态数组实现的List&#xff0c;支持快速随机访问。LinkedList: 基于链表实现的List&#xff0c;支持快速的插入和删除操作。Vector: 一个线程安全的动态数组…

基于Java的流浪动物救助管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09; 代码参考数据库参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者&am…

Qt中实现页面切换的两种方式

文章目录 方式一 &#xff1a;使用QStackedWidget讲解代码结构main.cpp完整代码运行结果&#xff1a; 方式二 &#xff1a;代码结构完整代码mainwindow.hnewmainwindow.hmain.cppmainwindow.cppnewmainwindow.cppmainwindow.uinewmainwindow.ui 效果 方式一 &#xff1a;使用QS…

docker 部署 若依 Ruoyi springboot+vue分离版 dockerCompose

本篇从已有虚拟机/服务器 安装好dokcer为基础开始讲解 1.部署mysql 创建conf data init三个文件夹 conf目录存放在mysql配置文件 init目录存放着若依数据库sql文件&#xff08;从navicat导出的并非若依框架自带sql&#xff09; 创建一个属于本次若依部署的网段&#xff08;只…

sql-50练习题6-10

sql练习题6-10题 前言数据库表结构介绍学生表课程表成绩表教师表 0-6 查询"李"姓老师的数量0-7 查询学过"李四"老师授课的同学的信息0-8 查询没学过"李四"老师授课的同学的信息0-9 查询学过编号为"01"并且也学过编号为"02"的…

Redis(09)| Reactor模式

我们在使用Redis的时候&#xff0c;通常是多个客户端连接Redis服务器&#xff0c;然后各自发送命令请求(例如Get、Set)到Redis服务器&#xff0c;最后Redis处理这些请求返回结果。 从上一篇博文《Redis&#xff08;08&#xff09;| 线程模型》中知道Redis是单线程。Redis除了处…

数据结构——栈与队列

目录 1. 中缀表达式转换为后缀表达式 2. 括号匹配问题 3. 栈实现队列 4. 约瑟夫环 1. 中缀表达式转换为后缀表达式 【问题描述】 输入一个中缀表达式&#xff0c;表达式中有、-、*、/四种运算以及&#xff08;、&#xff09;&#xff0c;表达式中的其他符号为大写的字母。实…

QT webengine显示HTML简单示例

文章目录 参考示例1TestWebenqine.promainwindow.hmainwindow.cppmain.cpp效果 示例2 (使用setDevToolsPage函数)main.cpp效果 参考 QT webengine显示HTML简单示例 示例1 编译器 : Desktop Qt 5.15.2 MSVC2019 64bit编辑器: QtCreator代码: TestWebenqine.pro # TestWeben…

pycharm运行R语言脚本(win10环境下安装)

文章目录 简介1. pycharm安装插件2. 安装R语言解释器2.1下载安装包2.2具体安装过程 3.编辑环境变量4 检验是否安装成功&#xff1a;5.安装需要的library6.pycharm中配置安装好的R语言解释器 简介 pycharm 安装 R language for Intellij R language for Intellij 是一个插件&am…

震惊!原来BUG是这么理解的!什么是BUG?软件错误(BUG)的概念

较为官方的概念&#xff1a; 当且仅当规格说明是存在的并且正确&#xff0c;程序与规格说明之间的 不匹配才是错误。 当需求规格说明书没有提到的功能&#xff0c;判断标准以最终用户为准&#xff1a;当程序没有实现其最终用户合理预期的 功能要求时&#xff0c;就是软…