[原]淘宝SKU组合查询算法实现
<div id="cnblogs_post_body">前端有多少事情可以做,能做到多好。一直在关注各大公司UED方面的知识,他们也代表了前端的力量,而且也很乐意和大家分享,把应用到项目的知识归类整理,再写成博客搬到网上来,充实这前端的内容,也是为想追寻和学习的人提供了场所,为想接触到一些前沿的知识提供了去处,感谢有这么一群人。大的科技公司基本都有自己的前端部门或团队,在网上也能看到他们的动态,像淘宝、阿里巴巴、腾讯、百度等等。前段时间在淘宝UED官网上看到一篇SKU组合查询算法探索,当时看过以后只是感觉挺牛的,而且讲的很具体,实现步骤和代码都想说的很详细,几种算法以及算法的复杂度都很有深入的分析,挺佩服这种专研精神的,当时只是隐约的感觉到这个算法在解决电商的商品拆分属性选择中可能会用到,但是具体的实现细节也没进行尝试。
后来公司正好要做一个项目,而且用的就是淘宝商品数据结构,商品详情页是属性选择也和淘宝的很类似,当时就想到了那篇文章,于是有反复看来两三遍,试了一下上面说的第二种算法(已经给出了源码),实现起来也不麻烦,虽然例子中给出的第二种算法得到的结果只有商品数量,但是经过修改也可以得到商品的价格,本打算这样就可以直接用的项目中好了。但是在看到第二种算法的优化后(没有提供源码),就想按照这种方式来实现,也是最初萌发出来的想法一致。
第二种算法会有大量的组合,它是基于原始属性值的结果组合和递归,而不是基于结果集的。其实第二种算法的优化,基于结果集的算法实现起来也不麻烦,原理就是把结果集的SKU中key值进行更小拆分组合,把拆分和组合后的结果信息放到SKUResult里面,这样在初始化一次完成,后面的选择可以根据这个结果集使用。把组合范围减少到key里面,这样能够搜索范围避免递归,而且得到的每个小的组合属性值的结果有用信息很丰富,数量和价格都包括其中。
但是又过了一段时间以后,项目被搁浅了,也不知道以后能用上不能了,写的示例也搁了许久,再不拿出来晾晾估计都该长毛变味了。
示例如下
测试地址: http://jsfiddle.net/tianshaojie/aGggS/embedded/result/
主要JS代码实现如下
<div id="scid:f32c3428-b7e9-4f15-a8ea-c502c7ff2e88:486d904e-aab3-45a0-bd4d-7323778b8c98" class="wlWriterEditableSmartContent" style="float: none; margin: 0px; display: inline; padding: 0px;">//原始属性集var keys = [ , , ];//测试结果集var data = { "10;20;30": { price: 100, count: 1 }, "10;20;31": { price: 111, count: 2 }, "11;20;30": { price: 122, count: 1 }, "10;21;31": { price: 133, count: 2 }, "10;21;32": { price: 144, count: 9 }}//保存最后的组合结果信息var SKUResult = {};//获得对象的keyfunction getObjKeys(obj) { if (obj !== Object(obj)) throw new TypeError('Invalid object'); var keys = []; for (var key in obj) if (Object.prototype.hasOwnProperty.call(obj, key)) keys = key; return keys;}//把组合的key放入结果集SKUResultfunction add2SKUResult(key, sku) { if(SKUResult) {//SKU信息key属性&middot; SKUResult.count += sku.count; SKUResult.prices.push(sku.price); } else { SKUResult = { count : sku.count, prices : }; }}//对一条SKU信息进行拆分组合function combineSKU(skuKeyAttrs, cnum, sku) { var len = skuKeyAttrs.length; for(var i = 0; i < len; i++) { var key = skuKeyAttrs; for(var j = i+1; j < len; j++) { if(j + cnum <= len) { var tempArr = skuKeyAttrs.slice(j, j+cnum);//安装组合个数获得属性值&middot; var genKey = key + ";" + tempArr.join(";");//得到一个组合key add2SKUResult(genKey, sku); } } }}//初始化得到结果集function initSKU() { var i, j, skuKeys = getObjKeys(data); for(i = 0; i < skuKeys.length; i++) { var skuKey = skuKeys;//一条SKU信息key var sku = data;//一条SKU信息value var skuKeyAttrs = skuKey.split(";"); //SKU信息key属性值数组 var len = skuKeyAttrs.length; //对每个SKU信息key属性值进行拆分组合 for(j = 0; j < len; j++) {//单个属性值作为key直接放入SKUResult add2SKUResult(skuKeyAttrs, sku); //对本组SKU信息key属性进行组合,组合个数为j (j > 0 && j < len-1) && combineSKU(skuKeyAttrs, j, sku); } //结果集接放入SKUResult SKUResult = { count:sku.count, prices: } }}
页:
[1]