当前位置: 首页 > news >正文

JavaScript设计模式系列一:工厂模式

设计模式

设计模式(design pattern)概念:
是一套反复使用、思想成熟、经过分类和无数实战设计经验的总结。是为了代码可重用、可扩展、可解耦、更容易被人理解和保证代码可靠性。

设计模式共有23种,我今天先来了解一下工厂模式(Factory Pattern),其他的模式将会在后续的博客中陆续为大家讲解。


工厂模式

工厂模式,正像它的名字一样,可以像工厂一样生产出我们想要的东西,只要我们提供原料。在我们日常编写代码过程中,很可能,我们不知不觉中就已经使用过工厂模式。我们看一个简单的例子:

//工厂模式很基础的的一个例子
function createPerson (name,age) {
  var person = new Object();
  person.name = name;
  person.age = age;
  return person;
}

var Holz = createPerson ("Holz", "21");
console.log(Holz);
/*
{
age: "21",
name: "Holz"
}
*/
var Tom = createPerson ("Tom", "7");
console.log(Tom);
/*
{
age: "7",
name: "Tom"
}
*/

在这个函数中,我可以通过传入不同的参数,创建具有不同参数值得对象。可以看到,工厂模式其实很简单。

工厂模式(Factory Pattern)主要分为简单工厂模式和抽象工厂模式。

简单工厂模式

简单工厂(Simple Factory Pattern)模式主要用于创建同一类的对象,比如我们创建需要一个身份,老师或者学生。可以通过下面这个函数:

//简单工厂模式。
var simpleCreatePerson = function (type, person) {
return this[type](person);
}

simpleCreatePerson.prototype = {
  student: function (person) {
    var Astudent = new Object();
    Astudent.name = person.name;
    Astudent.age = person.age;
    return Astudent;
  },
  teacher: function (person) {
    var Ateacher = new Object();
    Ateacher.name = person.name;
    Ateacher.age = person.age;
    return Ateacher;
  }
}

var teacher = new simpleCreatePerson("teacher", {name:"郑老师", age:25 });
var student = new simpleCreatePerson("student", {name:"郑同学", age:21 });

console.log(teacher);
/*
this is a teacher
{name:"郑老师", age: 25}
*/
console.log(student);
/*
this is a teacher
{name:"郑同学", age: 25}
*/

我们可以传入不同的 type 比如"student"或“teacher”去创建不同的对象,然后再通过不同的person对象去给里面的属性赋值。无论是"student"或者"teacher"都是person,所以说简单工厂函数就是用来创建同一类对象的不同实例。

上面的代码通过原型_prototype_去添加两个函数,(关于原型的知识,不是一两句可以讲完的,如果不懂得话,大家可以到网上去搜索相关的资料进行了解)然后通过this[type]去选择不同的方法,如果这里不明白的话我可以解释一下,this是一个对象,指向当前函数(在JavaScript里函数是对象),然后this[属性名/方法名] 去调用内部的属性或方法,那么this[type] (person)就是调用对应的方法去实例化一个具体的类(student或teacher)。通过this[type] (person)去调用。
这就是简单工厂函数。

抽象工厂模式

抽象工厂模式 :与简单工厂函数不同的是,抽象工厂函数会先设计好接口,具体的实现在子类中进行。这样看概念有点模糊,我们来看一个例子:

我们定义一个父类,abstractCreatePerson也就是创建一个人,然后这个人有一个自我介绍的方法方法selfIntroduction,那么不同的人就有不同的介绍内容,这个时候我们就可以用到抽象工厂模式:

var abstractCreatePerson = function () {};

abstractCreatePerson.prototype = {
  selfIntroduction: function () {
    throw new Error("请先实例化此方法!");
  }
}

var student = Object.create(abstractCreatePerson.prototype);
student.selfIntroduction = function () {
  console.log("I am a sutdent, my name is holz!");
}
student.selfIntroduction();
/*
I am a sutdent, my name is holz!
*/
var teacher = Object.create(abstractCreatePerson.prototype);
teacher.selfIntroduction = function () {
  console.log("I am a teacher, my name is xxxx!");
}
teacher.selfIntroduction();
/*
I am a teacher, my name is xxxx!
*/

可以看到这段代码,父类 abstractCreatePerson 中先声明了一个方法,然后通过 Object.create() 方法去继承这个父类,(因为这样不会覆盖原有的原型链),然后我们再在子类中去实例化方法,不同的子类就有了不同的实例方法。
这就是工厂模式,他提高了我们代码的可重用性,降低了模块之间的耦合度。


总结

  • 工厂模式,正像它的名字一样,可以像工厂一样生产出我们想要的东西,只要我们提供原料
  • 工厂模式提高了我们代码的可重用性,降低了模块之间的耦合度
  • 工厂模式主要用于创建同一类的对象的不同实例

相关文章:

  • Swift 计算文本的size
  • 73.node.js开发错误——TypeError: Cannot set property 'XXX' of undefined
  • ubuntu安装wxpython库
  • golang学习笔记(1):安装helloworld
  • 第 9 章 Spring Security
  • 一个字符串转换类
  • Ubuntu 16.04 LTS国内快速更新源
  • 考过网工的心得
  • 8.3. Spring Cloud 相关的 application.properties 配置
  • 大数据与云计算学习:数据分析(二)
  • 3.4 usermod命令 3.5 用户密码管理 3.6 mkpasswd命令
  • 剑指offer 矩阵覆盖
  • 从0开始弄一个面向OC数据库--终结篇
  • C++ Exercises(一)
  • restful+springmvc+mybatis+ webservice 分布式架构
  • JavaScript 如何正确处理 Unicode 编码问题!
  • CentOS 7 修改主机名
  • Java比较器对数组,集合排序
  • Java多线程(4):使用线程池执行定时任务
  • Linux中的硬链接与软链接
  • Netty 4.1 源代码学习:线程模型
  • Python语法速览与机器学习开发环境搭建
  • Spring Boot MyBatis配置多种数据库
  • 类orAPI - 收藏集 - 掘金
  • 前端性能优化--懒加载和预加载
  • 适配iPhoneX、iPhoneXs、iPhoneXs Max、iPhoneXr 屏幕尺寸及安全区域
  • 树莓派 - 使用须知
  • 用jquery写贪吃蛇
  • 优秀架构师必须掌握的架构思维
  • 与 ConTeXt MkIV 官方文档的接驳
  • 远离DoS攻击 Windows Server 2016发布DNS政策
  • [Shell 脚本] 备份网站文件至OSS服务(纯shell脚本无sdk) ...
  • #LLM入门|Prompt#3.3_存储_Memory
  • #NOIP 2014#day.2 T1 无限网络发射器选址
  • #Spring-boot高级
  • (11)MSP430F5529 定时器B
  • (C语言)二分查找 超详细
  • (力扣)1314.矩阵区域和
  • (利用IDEA+Maven)定制属于自己的jar包
  • (十六)一篇文章学会Java的常用API
  • (转)创业家杂志:UCWEB天使第一步
  • (轉)JSON.stringify 语法实例讲解
  • @SentinelResource详解
  • [ 环境搭建篇 ] 安装 java 环境并配置环境变量(附 JDK1.8 安装包)
  • [Angular 基础] - 数据绑定(databinding)
  • [C#]C# OpenVINO部署yolov8图像分类模型
  • [C++11 多线程同步] --- 条件变量的那些坑【条件变量信号丢失和条件变量虚假唤醒(spurious wakeup)】
  • [CentOs7]iptables防火墙安装与设置
  • [FFmpeg学习]从视频中获取图片
  • [GPT]Andrej Karpathy微软Build大会GPT演讲(上)--GPT如何训练
  • [IE6 only]关于Flash/Flex,返回数据产生流错误Error #2032的解决方式
  • [iOS]-NSTimer与循环引用的理解
  • [LeetCode] 2.两数相加
  • [LeetCode]Max Points on a Line
  • [one_demo_14]一个简单的easyui的demo