规格模式 Specification Pattern
创始人
2024-02-11 01:57:34
0

Specification Pattern

不了解 SOLID 的可以扫一眼这个:SOLID,面向对象设计五大基本原则。

今天扫 SOLID JS 的时候,授课者对 OCP 的一个实现方式利用了 Specification Pattern,所以这里也进行一下补充学习。

不过讲道理来说,因为 JavaScript 是一个弱类型的语言,所以很难强制让子类去实现其他语言中 interface/abstract 的函数,这也是一个局限了。

使用案例为,假设需要实现一个过滤器(filter),最初的实现方式如下:

const COLOR = Object.freeze({RED: 'red',GREEN: 'green',BLUE: 'blue',
});const SIZE = Object.freeze({SMALL: 'small',MEDIUM: 'medium',LARGE: 'large',
});class Product {constructor(name, color, size) {this.name = name;this.color = color;this.size = size;}
}class ProductFilter {filterByColor(products, color) {return products.filter((p) => p.color === color);}filterBySize(products, size) {return products.filter((p) => p.size === size);}
}const apple = new Product('Apple', COLOR.GREEN, SIZE.SMALL);
const tree = new Product('Tree', COLOR.GREEN, SIZE.LARGE);
const house = new Product('House', COLOR.BLUE, SIZE.LARGE);const products = [apple, tree, house];const pf = new ProductFilter();console.log('Green products:');
for (const p of pf.filterByColor(products, COLOR.GREEN)) {console.log(` * ${p.name} is green.`);
}

可以看到之后如果要写其他的 filter,也是可以继续在 ProductFilter 实现的,不过这里有几个问题:

  1. 继续实现其他的 filter 就要修改现有的代码
  2. 每一次新的需求来了,都需要额外实现新需求,这会引起 state space explosion 的问题,毕竟 3 个变量的 AND 操作,就需要 7 个方法去进行实现(A, B, C, A+B, B+C, A+C, A+B+C)。

一个解决这个问题的方法是使用 specification pattern,修改代码如下:

// should be done through inheritance, but js doesn't reenforce inheritance
// class Specification {}class ColorSpecification {constructor(color) {this.color = color;}isSatisfied(item) {return item.color === this.color;}
}class SizeSpecification {constructor(size) {this.size = size;}isSatisfied(item) {return item.size === this.size;}
}class FilterSpecs {filter(items, spec) {return items.filter((x) => spec.isSatisfied(x));}
}const fs = new FilterSpecs();console.log('Green products with specs:');
for (const p of fs.filter(products, new ColorSpecification(COLOR.GREEN))) {console.log(` * ${p.name} is green.`);
}

对于 AND 操作的实现补充如下:

class AndSpecifications {constructor(...specs) {this.specs = specs;}isSatisfied(item) {return this.specs.every((x) => x.isSatisfied(item));}
}console.log('Large and green products:');
const specs = new AndSpecifications(new ColorSpecification(COLOR.GREEN),new SizeSpecification(SIZE.LARGE)
);for (const p of fs.filter(products, specs)) {console.log(` * ${p.name} is green and large.`);
}

使用 specification pattern 的优势就在于极大地减少了函数的实现,原本对于 AND 的操作,3 个条件就需要实现 7 个方法,而后如果有更多的需求,就需要些更多的方法去满足这些需求,对于 nnn 个需求来说,就会有 n+n−1,+...+1n + n-1, + ... + 1n+n−1,+...+1,也就是 n2n^2n2 个方法。。使用了 specification pattern 现在只需要实现 3+1 个,并且如果之后有其他的需求了,在实现对应的基类后,只需要调用 AndSpecifications 并将对应的 specs 传进去就行,换言之,只需要实现 n+1n+1n+1 个方法。

同样的,如果对 OR,XOR 操作也有需求,那么 specification pattern 就能省下大量的代码和开发时间。

Reference

  • Specification pattern

    wiki 上提供了不少的代码案例,包括 C#,Python,C++和 TS。

相关内容

热门资讯

铜陵有色已回购2951万股 金... 7月1日,铜陵有色(000630)发布公告,截至2025年6月30日,公司已回购2951万股,占总股...
国家发改委:国内汽、柴油价格每... 据国家发改委,根据近期国际市场油价变化情况,按照现行成品油价格形成机制,自2025年7月1日24时起...
募资用途初步披露,亿纬锂能递表... 6月30日晚间,亿纬锂能(维权)公告称,公司已于当日向香港联交所递交了发行境外上市外资股(H股)并在...
国内汽柴油价格7月1日24时上... 全国平均来看,92号汽油、95号汽油和0号柴油每升上调0.18元、0.19元和0.19元。 ...
领益智造累计回购3823万股 ... 7月1日,领益智造(002600)发布公告,截至2025年6月30日,累计回购3823万股,占总股本...
最高奖励2000元!福州公开征... 夏木阴阴,水韵悠长。作为一座有着2200多年建城史的文化名城,福州自建城以来就与水密不可分。千百年来...
欧洲银行股上半年历史性飞跃:斯... 欧洲银行业上半年交出了一份令市场瞩目的成绩单。斯托克600银行指数以29%的涨幅创下1997年以来最...
1—5月湖南规模工业企业效益延...   长沙晚报掌上长沙7月1日讯(全媒体记者 周丛笑 通讯员 杨雪)1日,省统计局公布数据:1—5月,...
龙虎榜|山河药辅涨停,开源证券... 7月1日,山河药辅涨停,日振幅值达13.61%,日换手率达32.06%,收盘价14.72元,成交额8...
钻石的量子革命:这颗“原子级眼...   炒股就看金麒麟分析师研报,权威,专业,及时,全面,助您挖掘潜力主题机会! 在上一篇文章中,我们...