商品SKU功能设计与优化

原来的商品SKU设计存在着两个问题,一个是SKU表设计上面比较固化,无法扩展。另一个是当修改了商品信息之后,商品SKU的ID会发生变化,由于购物车表和订单商品表都关联了商品SKU的ID,这样就会导致匹配不上。最近对这两个问题做了点优化,下面来聊聊优化的思路。

商品的SPU和SKU

首先我们来了解下商品SPU和SKU的概念,可能很多没有接触过电商的朋友都不了解。

  • SPU(Standard Product Unit ):指的是标准商品单位,商品信息聚合的最小单位,是一组可复用、易检索的标准化信息的集合,该集合描述了一个商品的特性;

  • SKU(Stock Keeping Unit):库存量单位,是物理上不可分割的最小存货单元。

举个例子:比如说现在有个手机商品叫小米8,小米8有不同的属性,比如有它有黑色和蓝色的,有32G和64G版本的。此时小米8就是一个SPU,而小米8黑色64G就是一个SKU。

商品的SKU设计

以前的设计

商品的SKU信息是存储在pms_sku_stock表中的,使用sp1、sp2、sp3这三个属性来存储商品的销售属性,这样做很不灵活,也难以扩展。

这种做法也带来了后续的问题,比如我们的购物车和订单都会需要存储销售属性,这样的话都会需要添加sp1、sp2、sp3的属性。

改进后的设计

由于商品的销售属性是动态的,没法确定到底有多少个,此时我们可以改用JSON格式来存储,在pms_sku_stock表中添加了sp_data字段。

sp_data存储的就是一个JSON数组,比如颜色为黑色,容量为32G的手机存储信息如下。

[
    {
        "key": "颜色",
        "value": "黑色"
    },
    {
        "key": "容量",
        "value": "32G"
    }
]

这样修改以后,在原来的购物车表oms_cart_item和订单商品表oms_order_item中就都可以用JSON格式来存储销售属性了,使用的是product_attr字段。

商品关联SKU的修改

以前的做法

商品的SKU信息作为商品的关联信息,在修改商品信息时会同时进行修改。以前的做法是直接删除该商品的所有SKU信息,再重新添加。这样就会导致商品SKU中的ID被修改,由于在购物车和订单商品中关联了商品SKU的ID,就会导致原来的ID失效的问题。下面是原来修改商品中SKU信息的代码。

/**
 * 商品管理Service实现类
 * Created by macro on 2018/4/26.
 */
@Service
public class PmsProductServiceImpl implements PmsProductService {
    
        @Override
        public int update(Long id, PmsProductParam productParam) {
            //省略若干代码...
            //删除该商品关联的SKU
            PmsSkuStockExample skuStockExample = new PmsSkuStockExample();
            skuStockExample.createCriteria().andProductIdEqualTo(id);
            skuStockMapper.deleteByExample(skuStockExample);
            handleSkuStockCode(productParam.getSkuStockList(),id);
            //插入传入的所有SKU
            relateAndInsertList(skuStockDao, productParam.getSkuStockList(), id);
        }
}

改进后的做法

首先我们需要和前端约定下,新增的商品SKU信息不传ID,要修改的商品SKU信息传ID,删除的直接不传SKU信息。然后我们可以根据传入的SKU信息来确定需要新增、修改、删除的SKU信息,这样就可以做到在更新商品SKU信息时,不改变原来商品SKU的ID了,具体流程如下。

具体代码实现如下:

/**
 * 商品管理Service实现类
 * Created by macro on 2018/4/26.
 */
@Service
public class PmsProductServiceImpl implements PmsProductService {
    private void handleUpdateSkuStockList(Long id, PmsProductParam productParam) {
        //当前的sku信息
        List<PmsSkuStock> currSkuList = productParam.getSkuStockList();
        //当前没有sku直接删除
        if(CollUtil.isEmpty(currSkuList)){
            PmsSkuStockExample skuStockExample = new PmsSkuStockExample();
            skuStockExample.createCriteria().andProductIdEqualTo(id);
            skuStockMapper.deleteByExample(skuStockExample);
            return;
        }
        //获取初始sku信息
        PmsSkuStockExample skuStockExample = new PmsSkuStockExample();
        skuStockExample.createCriteria().andProductIdEqualTo(id);
        List<PmsSkuStock> oriStuList = skuStockMapper.selectByExample(skuStockExample);
        //获取新增sku信息
        List<PmsSkuStock> insertSkuList = currSkuList.stream().filter(item->item.getId()==null).collect(Collectors.toList());
        //获取需要更新的sku信息
        List<PmsSkuStock> updateSkuList = currSkuList.stream().filter(item->item.getId()!=null).collect(Collectors.toList());
        List<Long> updateSkuIds = updateSkuList.stream().map(PmsSkuStock::getId).collect(Collectors.toList());
        //获取需要删除的sku信息
        List<PmsSkuStock> removeSkuList = oriStuList.stream().filter(item-> !updateSkuIds.contains(item.getId())).collect(Collectors.toList());
        handleSkuStockCode(insertSkuList,id);
        handleSkuStockCode(updateSkuList,id);
        //新增sku
        if(CollUtil.isNotEmpty(insertSkuList)){
            relateAndInsertList(skuStockDao, insertSkuList, id);
        }
        //删除sku
        if(CollUtil.isNotEmpty(removeSkuList)){
            List<Long> removeSkuIds = removeSkuList.stream().map(PmsSkuStock::getId).collect(Collectors.toList());
            PmsSkuStockExample removeExample = new PmsSkuStockExample();
            removeExample.createCriteria().andIdIn(removeSkuIds);
            skuStockMapper.deleteByExample(removeExample);
        }
        //修改sku
        if(CollUtil.isNotEmpty(updateSkuList)){
            for (PmsSkuStock pmsSkuStock : updateSkuList) {
                skuStockMapper.updateByPrimaryKeySelective(pmsSkuStock);
            }
        }

    }
}

总结

如果我们要在数据库中存储一些格式不固定的属性时,可以采用JSON的形式进行存储。对于关联属性的修改,可以通过一些逻辑操作来实现不改变原有ID的修改。

项目源码地址

https://github.com/macrozheng/mall

推荐阅读


欢迎关注,点个在看

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 电商ERP商品管理SKU结构的设计应考虑到商品属性、属性值、库存以及价格等因素。 首先,SKU需包含商品基本信息,如商品ID、商品名称、商品描述等。同时,每个SKU可以对应多种商品属性,因此还需添加属性和属性值。属性可以包括属性ID、属性名称等字段,属性值可以包括属性值ID、属性值名称以及对应的属性ID等字段,以方便SKU与属性的关联。 其次,SKU还需考虑库存管理。可以在SKU中添加库存字段,记录当前SKU的库存数量。此外,还可添加预警字段,当库存数量低于设定值时,可以通过预警系统提醒相关人员及时补充库存。 此外,价格管理也是SKU的重要一部分。可以在SKU中添加价格字段,记录当前SKU的价格。如果存在活动价和原价区分,可以添加活动价字段和原价字段,以便进行价格展示和计算。 在设计SKU结构时,还需考虑到商品SKU的关系。一个商品可以对应多个SKU,而一个SKU只能对应一个商品。因此可以在SKU中添加商品ID字段,将SKU商品进行关联。 总之,电商ERP商品管理SKU结构的设计应当综合考虑商品属性、属性值、库存以及价格等因素,以便实现商品属性管理、库存管理和价格管理等功能。 ### 回答2: 电商ERP商品管理中,SKU(Stock Keeping Unit)结构的设计应考虑以下几个方面: 1. 商品基本信息:包括商品ID、商品名称、商品描述等字段,用于唯一标识商品和展示商品基本信息。 2. SKU属性信息:不同SKU可能存在不同的属性,如颜色、尺寸、款式等,可以通过添加属性字段来描述SKU的不同属性。 3. SKU库存信息:包括SKU的库存数量、可售数量等字段,用于实时更新和管理SKU的库存情况。 4. SKU价格信息:包括单价、折扣、促销价等字段,用于管理SKU的定价和促销活动。 5. SKU销售属性:对于存在销售属性的SKU,如鞋子的尺码、颜色等,可以通过添加销售属性字段来记录每个SKU的销售属性组合,方便在下单时进行选择。 6. SKU关联商品信息:对于有关联关系的商品,如套装、推荐组合等,可以通过添加关联商品字段来建立SKU与其他商品的关联关系。 7. SKU图片信息:每个SKU可能有不同的展示图片,可以通过添加图片字段来关联SKU对应的图片信息,方便展示和管理。 8. SKU状态信息:记录SKU的状态,如上架、下架、缺货、预售等,方便管理SKU的上下架和库存状态。 综上所述,电商ERP商品管理中,SKU结构的设计应综合考虑商品基本信息、SKU属性、库存信息、价格信息、销售属性、关联商品、图片和状态等多个方面,以便有效管理和展示SKU的信息。 ### 回答3: 电商ERP商品管理SKU设计需要考虑到以下几个方面。 首先,SKU是指库存单位,它应该包括商品的唯一识别码以及与该商品相关的属性信息,以便于与商品进行关联。因此,SKU应该包含一列用来存储商品的唯一识别码,比如商品编号或条形码。此外,还可以添加其他列来存储商品的属性信息,例如颜色、尺码、款式等。 其次,SKU的结构应该考虑到商品的多样性和组合性。同一款商品可能有多个SKU,每个SKU对应于不同的属性组合。因此,可以使用多列来存储不同的属性信息,然后使用这些列的组合作为SKU的唯一识别码。例如,如果商品具有颜色和尺码两个属性,则可以添加两列分别存储颜色和尺码的信息。 此外,SKU还应该包含与库存相关的字段。例如,可以添加一列来存储该SKU对应的库存数量,以便实时跟踪商品的库存情况。还可以添加一列来存储该SKU对应的价格信息,以便查询和更新商品的价格。 最后,SKU设计还应该考虑到性能和扩展性的需求。如果SKU的数据量较大,可以考虑使用索引来优化查询性能。另外,可以添加其他辅助或字段来存储与SKU相关的数据,如销售记录、供应商信息等,以便于扩展和管理。 综上所述,电商ERP商品管理SKU设计应该综合考虑商品属性、库存信息、性能需求和扩展性,以便于准确、高效地管理和查询商品信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值