Skip to content
微信公众号

工厂模式

日常项目开发中,遇到 new class 的场景,要考虑是否可用工厂模式。

介绍

创建对象的一种方式。不用每次都亲自创建对象,而是通过一个既定的“工厂”来生产对象。

示例

现在你要得到一个汉堡,你是跟服务员要(买)一个,还是自己动手做一个?这个问题,服务员就是工厂方法,而动手做一个其实就是new A()。 另外从快餐店考虑,你想要提供一个汉堡,是让服务员(工厂方法)做出来(new A())给客户,还是让客户自己做一个汉堡?

从这个示例很容易理解工厂模式的用意,所有的设计模式都是很讲道理的,很容易理解

伪代码

OOP 中,默认创建对象一般是 new class ,但一些情况下用 new class 会很不方便。

js
// 伪代码
let f1
class Foo {}

if (a) {
    f1 = Foo(x)
}
if (b) {
    f2 = Foo(x, y)
}

此时就需要一个“工厂”,把创建者和 class 分离,符合开放封闭原则。

js
// 工厂
function create(a, b) {
    if (a) {
        return Foo(x)
    }
    if (b) {
        return Foo(x, y)
    }
}

const f1 = create(a, b)

注意

工厂模式可以拆分为三个:

  • 工厂方法模式
  • 抽象工厂模式
  • 建造者模式

前端用不到这么细致,只需要掌握核心的工厂模式即可。

标准的工厂模式

ts
interface IProduct {
    name: string
    fn1: () => void
    fn2: () => void
}

class Product1 implements IProduct { 
    name: string
    constructor(name: string) {
        this.name = name
    }
    fn1() { 
        alert('product1 fn1')
    }
    fn2() { 
        alert('product1 fn2')
    }
}

class Product2 implements IProduct { 
    name: string
    constructor(name: string) {
        this.name = name
    }
    fn1() { 
        alert('product2 fn1')
    }
    fn2() { 
        alert('product2 fn2')
    }
}

class Creator { 
    create(type: string, name: string): IProduct {
        if (type === 'p1') {
            return new Product1(name)
        }
        if (type === 'p2') {
            return new Product2(name)
        }
        throw new Error('Invalid type')
    }
}

简单的工厂模式

ts
class Product { 
    name: string
    constructor(name: string) {
        this.name = name
    }
    fn1() { 
        alert('product fn1')
    }
    fn2() { 
        alert('product fn2')
    }
}


class Creator { 
    create(name: string): Product {
        return new Product(name)
    }
}

是否符合设计原则?

5 大设计原则中,最重要的就是:开放封闭原则,对扩展开放,对修改封闭

  • 工厂和类分离,解耦
  • 可以扩展多个类
  • 工厂的创建逻辑也可以自由扩展

场景

jQuery $('div')

ts
// 扩展 window.$
declare interface Window { 
    $: (selector: string) => JQuery
}

class JQuery {
    selector: string
    length: number

    constructor(selector: string) {
        const domList = Array.prototype.slice.call(document.querySelectorAll(selector))
        const length = domList.length
        for (let i = 0; i < length; i++) { 
            this[i] = domList[0]
        }

        this.selector = selector
        this.length = length
    }

    append(elem: HTMLElement): JQuery {
        // ...
        return this
    }

    addClass(key: string, value: string): JQuery {
        // ...
        return this
    }

    html(htmlStr: string): JQuery | string { 
        if (htmlStr) {
            // set html
            return this
        } else { 
            // get html
            const html = 'xxx'
            return html
        }
    }
}

window.$ = (selector) => { 
    return new JQuery(selector)
}

做一个对比,如果开放给用户的不是$,然后让用户自己去new JQuery(selector),带来的问题:

  • 不方便链式操作,如$('div').append($('#p1')).html()
  • 不宜将构造函数暴露给用户,尽量高内聚、低耦合

Vue _createElementVNode

在线编译 https://vue-next-template-explorer.netlify.app/

html
<div>
  <span>静态文字</span>
  <span :id="hello" class="bar">{{ msg }}</span>
</div>

会编译出很多 _createXxx JS 代码。这些就是工厂函数,创建 vnode 。

js
export function render(_ctx, _cache, $props, $setup, $data, $options) {
  return (_openBlock(), _createElementBlock("div", null, [
    _createElementVNode("span", null, "静态文字"),
    _createElementVNode("span", {
      id: _ctx.hello,
      class: "bar"
    }, _toDisplayString(_ctx.msg), 9 /* TEXT, PROPS */, ["id"])
  ]))
}

React createElement

在线编译 https://www.babeljs.cn/repl

在 React 中使用 JSX 语法

jsx
const profile = <div>
  <img src="avatar.png" className="profile" />
  <h3>{[user.firstName, user.lastName].join(' ')}</h3>
</div>

这是一种语法糖,编译之后就会是

js
// 返回 vnode
const profile = React.createElement("div", null,
    React.createElement("img", { src: "avatar.png", className: "profile" }),
    React.createElement("h3", null, [user.firstName, user.lastName].join(" "))
);

其实React.createElement也是一个工厂,模拟代码

js
class Vnode(tag, attrs, children) {
    // ...省略内部代码...
}
React.createElement =  function (tag, attrs, children) {
    return new Vnode(tag, attrs, children)
}

本站总访问量次,本站总访客数人次
Released under the MIT License.