【学习目标】掌握策略模式核心思想基于IGoodsComparator接口封装排序规则实现排序逻辑的灵活扩展与解耦理解工厂模式的应用场景开发排序工厂类统一管理比较器实例屏蔽底层实现细节整合单例管控策略模式工厂模式构建“数据统一、规则可扩展、调用低耦合”的商品排序体系实战验证多场景排序能力实现商品列表页“一键切换排序规则”完成从功能实现到工程化设计的进阶。【学习重点】策略核心以接口为契约将库存、价格升/降序等排序规则封装为独立策略类遵循“开闭原则”工厂实现通过排序工厂类统一创建比较器实例页面仅需传入排序类型即可调用降低耦合架构整合将策略模式、工厂模式融入上一节的单例体系保证全局数据一致性的同时提升扩展性实战验证在商品列表页新增下拉选择器支持多规则切换验证工程化重构的优势。一、工程结构复制上一节的ClassObjectDemo_6工程重命名为ClassObjectDemo_7新增排序策略与工厂相关文件ClassObjectDemo_7/ ├── entry/ # 应用主模块 │ ├── src/main/ets/ │ │ ├── pages/ # 视图层 │ │ │ └── Index.ets # 核心演示页排序规则切换实战 │ │ ├── model/ # 核心模型层 │ │ │ ├── entity/ # 实体层复用原有 │ │ │ ├── interface/ # 契约层复用IGoodsComparator接口 │ │ │ ├── comparator/ # 排序策略实现新增价格降序比较器 │ │ │ └── utils/ # 工具层新增排序工厂类 │ │ │ ├── GoodsList.ets # 复用泛型商品容器 │ │ │ ├── GoodsTool.ets # 复用泛型商品工具类 │ │ │ ├── GoodsManager.ets # 复用单例商品管理中枢 │ │ │ └── SortFactory.ets # 新增排序比较器工厂类 │ ├── resources/ # 资源目录 │ └── module.json5 # 模块配置 └── hvigorfile.ts # 构建脚本二、核心痛点回顾与技术选型2.1 第十六节排序体系的核心问题在第十六节的实现中页面调用排序逻辑需要直接依赖具体比较器类// 原页面调用方式强耦合具体实现this.goodsListGoodsTool.sortByComparator(this.goodsList,newStockAscComparator());存在两个核心问题耦合度高页面需导入所有用到的比较器若比较器类名/路径变更所有调用处都需修改扩展繁琐例如新增价格降序排序时需创建PriceDescComparator再在页面导入并实例化规则越多越混乱。2.2 技术选型策略模式工厂模式技术特性解决的核心问题适用场景策略模式排序规则与页面解耦新增规则无需修改页面多规则灵活切换的场景工厂模式统一管理比较器实例屏蔽创建细节实例化逻辑复杂、规则较多单例模式保证商品数据全局唯一排序后数据同步全应用商品数据统一管控三、策略模式排序规则的解耦与扩展3.1 策略模式核心逻辑策略模式的核心是接口定义契约类实现具体策略核心优势新增策略无需修改原有代码符合开闭原则对扩展开放、对修改关闭策略类专注单一排序规则职责清晰便于维护页面仅依赖接口不依赖具体实现降低耦合。3.2 新增价格降序排序策略基于IGoodsComparator接口新增价格降序比较器与原有库存、价格升序比较器共同构成策略族// model/comparator/PriceDescComparator.ets价格降序import{AbstractGoods}from../entity/AbstractGoods;import{IGoodsComparator}from../interface/behavior/IGoodsComparator;/** * 价格降序比较器 */exportclassPriceDescComparatorimplementsIGoodsComparatorAbstractGoods{compare(a:AbstractGoods,b:AbstractGoods):number{returnb.price-a.price;}}原有StockAscComparator/StockDescComparator/PriceAscComparator保持不变共同构成排序策略库。四、工厂模式排序比较器的统一管理4.1 工厂模式核心逻辑排序工厂类的核心是封装比较器的实例化逻辑对外提供统一的获取接口页面只需传入排序类型无需关注具体实现类同时通过缓存复用无状态的比较器实例减少内存开销。4.2 实现SortFactory排序工厂// model/utils/SortFactory.etsimport{IGoodsComparator}from../interface/behavior/IGoodsComparator;import{AbstractGoods}from../entity/AbstractGoods;import{StockAscComparator}from../comparator/StockAscComparator;import{StockDescComparator}from../comparator/StockDescComparator;import{PriceAscComparator}from../comparator/PriceAscComparator;import{PriceDescComparator}from../comparator/PriceDescComparator;// 新增价格降序导入// 定义排序类型枚举exportenumSortType{STOCK_ASCstockAsc,// 库存升序STOCK_DESCstockDesc,// 库存降序PRICE_ASCpriceAsc,// 价格升序PRICE_DESCpriceDesc// 新增价格降序}/** * 排序比较器工厂类 * 核心根据排序类型返回对应比较器实例屏蔽创建细节缓存实例减少重复创建 */exportclassSortFactory{// 缓存比较器实例无状态策略类复用提升性能privatestaticcomparatorCache:MapSortType,IGoodsComparatorAbstractGoodsnewMap();/** * 获取比较器实例 * param sortType 排序类型 * returns 对应的比较器实例 */publicstaticgetComparator(sortType:SortType):IGoodsComparatorAbstractGoods{// 从缓存中读取减少每次创建新的排序比较器如果不存在则创建新的拍下比较器并存到Map中if(!SortFactory.comparatorCache.has(sortType)){switch(sortType){caseSortType.STOCK_ASC:SortFactory.comparatorCache.set(sortType,newStockAscComparator());break;caseSortType.STOCK_DESC:SortFactory.comparatorCache.set(sortType,newStockDescComparator());break;caseSortType.PRICE_ASC:SortFactory.comparatorCache.set(sortType,newPriceAscComparator());break;caseSortType.PRICE_DESC:SortFactory.comparatorCache.set(sortType,newPriceDescComparator());break;default:SortFactory.comparatorCache.set(sortType,newStockAscComparator());}}returnSortFactory.comparatorCache.get(sortType)!;}}五、工程实战排序规则的灵活切换5.1 页面核心逻辑// pages/Index.etsimport{AbstractGoods}from../model/entity/AbstractGoods;import{DigitalGoods}from../model/entity/DigitalGoods;import{BookGoods}from../model/entity/BookGoods;import{GoodsStatus}from../model/entity/GoodsStatus;import{GoodsManager}from../model/utils/GoodsManager;import{GoodsTool}from../model/utils/GoodsTool;import{SortFactory,SortType}from../model/utils/SortFactory;import{IBookGoods}from../model/interface/attribute/IBookGoods;import{IDigitalGoods}from../model/interface/attribute/IDigitalGoods;EntryComponentstruct Index{StategoodsList:AbstractGoods[][];StatecurrentSortKey:SortTypeSortType.STOCK_ASC;StatecurrentSortValue:string库存升序;StateselectMenus:SelectOption[][{value:库存升序},{value:库存降序},{value:价格升序},{value:价格降序}];privategoodsManagerGoodsManager.getInstance();// 排序规则配置表Map数组方式适配Select索引联动privatesortMapList:Mapstring,SortType[][newMap([[库存升序,SortType.STOCK_ASC]]),newMap([[库存降序,SortType.STOCK_DESC]]),newMap([[价格升序,SortType.PRICE_ASC]]),newMap([[价格降序,SortType.PRICE_DESC]])];aboutToAppear(){this.initData();this.sortGoods();}// 初始化商品数据privateinitData():void{constdigitalProps:IDigitalGoods{name:鸿蒙Mate70手机,price:5999,stock:100,costPrice:4000,category:数码产品,status:GoodsStatus.ON_SHELF,brand:华为,warranty:2};consttabletProps:IDigitalGoods{name:鸿蒙平板Pro,price:3999,stock:80,costPrice:2500,category:数码产品,status:GoodsStatus.ON_SHELF,brand:华为,warranty:2};constbookProps:IBookGoods{name:鸿蒙开发实战,price:69,stock:500,costPrice:30,category:图书,status:GoodsStatus.ON_SHELF,author:散修,publishDate:2025-01};constgoodsList[newDigitalGoods(digitalProps),newDigitalGoods(tabletProps),newBookGoods(bookProps)];this.goodsManager.addGoodsBatch(goodsList);this.goodsListthis.goodsManager.getGoodsList();}// 排序商品列表基于工厂获取比较器解耦具体实现privatesortGoods():void{// 共排序工厂中获取对应的排序比较实例constcomparatorSortFactory.getComparator(this.currentSortKey);this.goodsListGoodsTool.sortByComparator(this.goodsList,comparator);// 控制台日志输出排序规则结果便于调试console.log([排序切换] 当前规则${this.currentSortValue});console.log([排序结果],this.goodsList.map(g${g.name}价格¥${g.price}| 库存${g.stock}));}build(){Column(){Text(工厂策略模式 排序演示).fontSize(20).fontWeight(FontWeight.Bold).margin(20);// 排序选择器Row({space:10}){Text(排序规则).fontSize(16);Select(this.selectMenus).value(this.currentSortValue).onSelect((index:number,key:string){this.currentSortValuekey;constsortTypethis.sortMapList[index].get(key);this.currentSortKeysortTypeasSortType;this.sortGoods();}).width(200);}.margin({bottom:15});// 商品列表空状态兜底提升用户体验Scroll(){Column({space:10}){if(this.goodsList.length0){Text(暂无商品数据).fontSize(16).fontColor(#999).margin(20);}else{ForEach(this.goodsList,(goods:AbstractGoods){Column({space:5}){Text(goods.name).fontSize(18).fontWeight(FontWeight.Medium);Text(分类${goods.category}| 价格¥${goods.price.toFixed(2)}| 库存${goods.stock}件).fontSize(14).fontColor(#666);if(goodsinstanceofDigitalGoods){Text(品牌${goods.brand}| 质保${goods.warranty}年).fontSize(12).fontColor(#999);}if(goodsinstanceofBookGoods){Text(作者${goods.author}| 出版日期${goods.publishDate}).fontSize(12).fontColor(#999);}}.width(90%).padding(15).backgroundColor(Color.White).borderRadius(8);})}}.padding(10).width(100%);}.layoutWeight(1).width(100%);}.width(100%).height(100%).backgroundColor(#f5f5f5).padding(10);}}5.2 演示效果六、架构整合与工程化最佳实践6.1 三层架构整合逻辑本次重构后商品管理体系形成“数据层-规则层-调用层”的三层架构各层职责清晰、解耦单例GoodsManager全局数据层- 统一管控商品数据- 保证数据唯一性GoodsTool工具层- 排序/筛选等通用能力- 依赖策略接口SortFactory工厂层- 封装比较器实例化- 屏蔽底层实现细节- 缓存策略实例排序策略族规则层- IGoodsComparator定契约- 各类实现具体排序规则- 无状态可复用页面调用层- 仅依赖工厂/枚举- 无需关注规则实现- 一键切换排序规则6.2 工程化最佳实践策略模式应用原则当同一类业务有多种实现方案时优先用接口定义契约避免if-else分支判断新增策略只需实现接口无需修改原有代码符合开闭原则策略类设计为无状态便于缓存复用如比较器类无属性仅含compare方法。工厂模式应用原则当实例化逻辑复杂或需要统一管理时使用工厂类封装通过枚举限定入参类型避免字符串硬编码导致的错误无状态策略类通过缓存复用减少内存开销和实例化耗时。与单例模式的配合单例负责数据的全局唯一性策略和工厂负责业务逻辑的扩展各司其职排序操作基于单例数据保证多页面/多组件排序后数据同步单例类仅存储核心数据页面临时状态如当前排序值保留在页面内部避免内存泄漏。七、内容总结策略模式核心以IGoodsComparator接口为契约将排序规则封装为独立策略类如PriceDescComparator实现规则与页面解耦新增规则仅需新增策略类无需修改原有代码工厂模式核心通过SortFactory统一管理比较器实例化逻辑页面仅需传入SortType枚举即可调用屏蔽底层实现细节同时缓存无状态策略实例提升性能架构整合价值单例数据统一策略规则扩展工厂实例管理的整合既保证商品数据全局唯一又让排序体系具备“低耦合、高扩展”的工程化特性实战优化补充商品列表空状态兜底、排序日志增强、策略实例缓存提升代码健壮性和运行性能。八、代码仓库工程名称ClassObjectDemo_7代码地址https://gitee.com/juhetianxia321/harmony-os-code-base.git下节预告本节课我们通过工厂策略模式完成了商品排序体系的重构也掌握了“类实现接口”的核心用法但当业务场景升级需要为商品实现多种促销规则折扣、多档位满减、赠品、满折、会员专属优惠等时单纯依赖“一个类实现多个接口”的基础用法会暴露出致命问题商品类职责爆炸商品类会充斥大量促销计算逻辑违背“单一职责原则”代码臃肿且难以维护规则叠加混乱多种促销规则的叠加顺序如先满减后折扣/先折扣后满减、优先级如400-40优先于200-20无法统一管控易出现计算错误扩展成本指数级增加新增第三种促销规则时需让商品类实现新接口所有调用处都要同步调整完全违背“开闭原则”代码复用性差不同商品的同类促销规则如数码/图书都支持满200减20无法复用同一套实现逻辑重复代码大量滋生。下一节我们将跳出“单纯用类实现接口”的思维局限通过进阶版策略模式破解多促销规则的扩展难题