设计模式之组合模式(营销差异化人群发券,决策树引擎搭建场景)
前言:
往往很多大需求都是通过增删改查堆出来的,今天要一个需求if一下,明天加个内容else扩展一下。日积月累需求也就越来越大,扩展和维护的成本也就越来越高。往往大部分研发是不具备产品思维和整体业务需求导向的,总以为写好代码完成功能即可。但这样的不考虑扩展性的实现,很难让后续的需求都快速迭代,久而久之就会被陷入恶性循环,每天都有bug要改。
从上图可以看到这有点像螺丝🔩和螺母,通过一堆的链接组织出一棵结构树。而这种通过把相似对象(也可以称作是方法)组合成一组可被调用的结构树对象的设计思路叫做组合模式。
这种设计方式可以让你的服务组节点进行自由组合对外提供服务,例如你有三个原子校验功能(A:身份证、B:银行卡、C:手机号)服务并对外提供调用使用。有些调用方需要使用AB组合,有些调用方需要使用到CBA组合,还有一些可能只使用三者中的一个。那么这个时候你就可以使用组合模式进行构建服务,对于不同类型的调用方配置不同的组织关系树,而这个树结构你可以配置到数据库中也可以不断的通过图形界面来控制树结构。
用一坨坨代码实现
公司里要都是这样的程序员绝对省下不少成本,根本不要搭建微服务,一个工程搞定所有业务!
package oom.lm.design;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;public class EngineController {private Logger logger = LoggerFactory.getLogger(EngineController.class);public String process(final String userId, final String userSex, final int userAge) {logger.info("ifelse实现方式判断用户结果。userId:{} userSex:{} userAge:{}", userId, userSex, userAge);if ("man".equals(userSex)) {if (userAge < 25) {return "果实A";}if (userAge >= 25) {return "果实B";}}if ("woman".equals(userSex)) {if (userAge < 25) {return "果实C";}if (userAge >= 25) {return "果实D";}}return null;}}
除了我们说的扩展性和每次的维护以外,这样的代码实现起来是最快的。而且从样子来看也很适合新人理解。
但是我劝你别写,写这样代码不是被扣绩效就是被开除。
package oom.lm.design.test;import oom.lm.design.EngineController;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;public class ApiTest {private Logger logger = LoggerFactory.getLogger(ApiTest.class);@Testpublic void test_EngineController() {EngineController engineController = new EngineController();String process = engineController.process("Oli09pLkdjh", "man", 29);logger.info("测试结果:{}", process);}}
组合模式重构代码
组合模式模型结构
树节点逻辑过滤器接口
package oom.lm.design.domain.service.logic;import oom.lm.design.domain.model.vo.TreeNodeLink;import java.util.List;
import java.util.Map;/******/
public interface LogicFilter {/*** 逻辑决策器** @param matterValue 决策值* @param treeNodeLineInfoList 决策节点* @return 下一个节点Id*/Long filter(String matterValue, List<TreeNodeLink> treeNodeLineInfoList);/*** 获取决策值** @param decisionMatter 决策物料* @return 决策值*/String matterValue(Long treeId, String userId, Map<String, String> decisionMatter);}
决策抽象类提供基础服务
package oom.lm.design.domain.service.logic;import oom.lm.design.domain.model.vo.TreeNodeLink;import java.util.List;
import java.util.Map;/******/
public abstract class BaseLogic implements LogicFilter {@Overridepublic Long filter(String matterValue, List<TreeNodeLink> treeNodeLinkList) {for (TreeNodeLink nodeLine : treeNodeLinkList) {if (decisionLogic(matterValue, nodeLine)){return nodeLine.getNodeIdTo();}}return 0L;}@Overridepublic abstract String matterValue(Long treeId, String userId, Map<String, String> decisionMatter);private boolean decisionLogic(String matterValue, TreeNodeLink nodeLink) {switch (nodeLink.getRuleLimitType()) {case 1:return matterValue.equals(nodeLink.getRuleLimitValue());case 2:return Double.parseDouble(matterValue) > Double.parseDouble(nodeLink.getRuleLimitValue());case 3:return Double.parseDouble(matterValue) < Double.parseDouble(nodeLink.getRuleLimitValue());case 4:return Double.parseDouble(matterValue) >= Double.parseDouble(nodeLink.getRuleLimitValue());case 5:return Double.parseDouble(matterValue) <= Double.parseDouble(nodeLink.getRuleLimitValue());default:return false;}}}
树节点逻辑实现类
package oom.lm.design.domain.service.logic.impl;import oom.lm.design.domain.service.logic.BaseLogic;import java.util.Map;public class UserAgeFilter extends BaseLogic {@Overridepublic String matterValue(Long treeId, String userId, Map<String, String> decisionMatter) {return decisionMatter.get("age");}}
package oom.lm.design.domain.service.logic.impl;import oom.lm.design.domain.service.logic.BaseLogic;import java.util.Map;public class UserGenderFilter extends BaseLogic {@Overridepublic String matterValue(Long treeId, String userId, Map<String, String> decisionMatter) {return decisionMatter.get("gender");}}
以上两个决策逻辑的节点获取值的方式都非常简单,只是获取用户的入参即可。实际的业务开发可以从数据库、RPC接口、缓存运算等各种方式获取。
决策引擎接口定义
package oom.lm.design.domain.service.engine;import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;import java.util.Map;public interface IEngine {EngineResult process(final Long treeId, final String userId, TreeRich treeRich, final Map<String, String> decisionMatter);}
决策节点配置
package oom.lm.design.domain.service.engine;import oom.lm.design.domain.service.logic.LogicFilter;
import oom.lm.design.domain.service.logic.impl.UserAgeFilter;
import oom.lm.design.domain.service.logic.impl.UserGenderFilter;import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;public class EngineConfig {static Map<String, LogicFilter> logicFilterMap;static {logicFilterMap = new ConcurrentHashMap<>();logicFilterMap.put("userAge", new UserAgeFilter());logicFilterMap.put("userGender", new UserGenderFilter());}public Map<String, LogicFilter> getLogicFilterMap() {return logicFilterMap;}public void setLogicFilterMap(Map<String, LogicFilter> logicFilterMap) {this.logicFilterMap = logicFilterMap;}}
基础决策引擎功能
package oom.lm.design.domain.service.engine;import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;
import oom.lm.design.domain.model.vo.TreeNode;
import oom.lm.design.domain.model.vo.TreeRoot;
import oom.lm.design.domain.service.logic.LogicFilter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.util.Map;
public abstract class EngineBase extends EngineConfig implements IEngine {private Logger logger = LoggerFactory.getLogger(EngineBase.class);@Overridepublic abstract EngineResult process(Long treeId, String userId, TreeRich treeRich, Map<String, String> decisionMatter);protected TreeNode engineDecisionMaker(TreeRich treeRich, Long treeId, String userId, Map<String, String> decisionMatter) {TreeRoot treeRoot = treeRich.getTreeRoot();Map<Long, TreeNode> treeNodeMap = treeRich.getTreeNodeMap();// 规则树根IDLong rootNodeId = treeRoot.getTreeRootNodeId();TreeNode treeNodeInfo = treeNodeMap.get(rootNodeId);//节点类型[NodeType];1子叶、2果实while (treeNodeInfo.getNodeType().equals(1)) {String ruleKey = treeNodeInfo.getRuleKey();LogicFilter logicFilter = logicFilterMap.get(ruleKey);String matterValue = logicFilter.matterValue(treeId, userId, decisionMatter);Long nextNode = logicFilter.filter(matterValue, treeNodeInfo.getTreeNodeLinkList());treeNodeInfo = treeNodeMap.get(nextNode);logger.info("决策树引擎=>{} userId:{} treeId:{} treeNode:{} ruleKey:{} matterValue:{}", treeRoot.getTreeName(), userId, treeId, treeNodeInfo.getTreeNodeId(), ruleKey, matterValue);}return treeNodeInfo;}}
这里主要提供决策树流程的处理过程,有点像通过链路的关系(性别、年龄)在二叉树中寻找果实节点的过程。
决策引擎的实现
package oom.lm.design.domain.service.engine.impl;import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;
import oom.lm.design.domain.model.vo.TreeNode;
import oom.lm.design.domain.service.engine.EngineBase;import java.util.Map;public class TreeEngineHandle extends EngineBase {@Overridepublic EngineResult process(Long treeId, String userId, TreeRich treeRich, Map<String, String> decisionMatter) {// 决策流程TreeNode treeNode = engineDecisionMaker(treeRich, treeId, userId, decisionMatter);// 决策结果return new EngineResult(userId, treeId, treeNode.getTreeNodeId(), treeNode.getNodeValue());}}
测试验证
package oom.lm.design.test;import com.alibaba.fastjson.JSON;
import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;
import oom.lm.design.domain.model.vo.TreeNode;
import oom.lm.design.domain.model.vo.TreeNodeLink;
import oom.lm.design.domain.model.vo.TreeRoot;
import oom.lm.design.domain.service.engine.IEngine;
import oom.lm.design.domain.service.engine.impl.TreeEngineHandle;
import org.junit.Before;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;public class ApiTest {private Logger logger = LoggerFactory.getLogger(ApiTest.class);private TreeRich treeRich;@Beforepublic void init() {// 节点:1TreeNode treeNode_01 = new TreeNode();treeNode_01.setTreeId(10001L);treeNode_01.setTreeNodeId(1L);treeNode_01.setNodeType(1);treeNode_01.setNodeValue(null);treeNode_01.setRuleKey("userGender");treeNode_01.setRuleDesc("用户性别[男/女]");// 链接:1->11TreeNodeLink treeNodeLink_11 = new TreeNodeLink();treeNodeLink_11.setNodeIdFrom(1L);treeNodeLink_11.setNodeIdTo(11L);treeNodeLink_11.setRuleLimitType(1);treeNodeLink_11.setRuleLimitValue("man");// 链接:1->12TreeNodeLink treeNodeLink_12 = new TreeNodeLink();treeNodeLink_12.setNodeIdFrom(1L);treeNodeLink_12.setNodeIdTo(12L);treeNodeLink_12.setRuleLimitType(1);treeNodeLink_12.setRuleLimitValue("woman");List<TreeNodeLink> treeNodeLinkList_1 = new ArrayList<>();treeNodeLinkList_1.add(treeNodeLink_11);treeNodeLinkList_1.add(treeNodeLink_12);treeNode_01.setTreeNodeLinkList(treeNodeLinkList_1);// 节点:11TreeNode treeNode_11 = new TreeNode();treeNode_11.setTreeId(10001L);treeNode_11.setTreeNodeId(11L);treeNode_11.setNodeType(1);treeNode_11.setNodeValue(null);treeNode_11.setRuleKey("userAge");treeNode_11.setRuleDesc("用户年龄");// 链接:11->111TreeNodeLink treeNodeLink_111 = new TreeNodeLink();treeNodeLink_111.setNodeIdFrom(11L);treeNodeLink_111.setNodeIdTo(111L);treeNodeLink_111.setRuleLimitType(3);treeNodeLink_111.setRuleLimitValue("25");// 链接:11->112TreeNodeLink treeNodeLink_112 = new TreeNodeLink();treeNodeLink_112.setNodeIdFrom(11L);treeNodeLink_112.setNodeIdTo(112L);treeNodeLink_112.setRuleLimitType(4);treeNodeLink_112.setRuleLimitValue("25");List<TreeNodeLink> treeNodeLinkList_11 = new ArrayList<>();treeNodeLinkList_11.add(treeNodeLink_111);treeNodeLinkList_11.add(treeNodeLink_112);treeNode_11.setTreeNodeLinkList(treeNodeLinkList_11);// 节点:12TreeNode treeNode_12 = new TreeNode();treeNode_12.setTreeId(10001L);treeNode_12.setTreeNodeId(12L);treeNode_12.setNodeType(1);treeNode_12.setNodeValue(null);treeNode_12.setRuleKey("userAge");treeNode_12.setRuleDesc("用户年龄");// 链接:12->121TreeNodeLink treeNodeLink_121 = new TreeNodeLink();treeNodeLink_121.setNodeIdFrom(12L);treeNodeLink_121.setNodeIdTo(121L);treeNodeLink_121.setRuleLimitType(3);treeNodeLink_121.setRuleLimitValue("25");// 链接:12->122TreeNodeLink treeNodeLink_122 = new TreeNodeLink();treeNodeLink_122.setNodeIdFrom(12L);treeNodeLink_122.setNodeIdTo(122L);treeNodeLink_122.setRuleLimitType(4);treeNodeLink_122.setRuleLimitValue("25");List<TreeNodeLink> treeNodeLinkList_12 = new ArrayList<>();treeNodeLinkList_12.add(treeNodeLink_121);treeNodeLinkList_12.add(treeNodeLink_122);treeNode_12.setTreeNodeLinkList(treeNodeLinkList_12);// 节点:111TreeNode treeNode_111 = new TreeNode();treeNode_111.setTreeId(10001L);treeNode_111.setTreeNodeId(111L);treeNode_111.setNodeType(2);treeNode_111.setNodeValue("果实A");// 节点:112TreeNode treeNode_112 = new TreeNode();treeNode_112.setTreeId(10001L);treeNode_112.setTreeNodeId(112L);treeNode_112.setNodeType(2);treeNode_112.setNodeValue("果实B");// 节点:121TreeNode treeNode_121 = new TreeNode();treeNode_121.setTreeId(10001L);treeNode_121.setTreeNodeId(121L);treeNode_121.setNodeType(2);treeNode_121.setNodeValue("果实C");// 节点:122TreeNode treeNode_122 = new TreeNode();treeNode_122.setTreeId(10001L);treeNode_122.setTreeNodeId(122L);treeNode_122.setNodeType(2);treeNode_122.setNodeValue("果实D");// 树根TreeRoot treeRoot = new TreeRoot();treeRoot.setTreeId(10001L);treeRoot.setTreeRootNodeId(1L);treeRoot.setTreeName("规则决策树");Map<Long, TreeNode> treeNodeMap = new HashMap<>();treeNodeMap.put(1L, treeNode_01);treeNodeMap.put(11L, treeNode_11);treeNodeMap.put(12L, treeNode_12);treeNodeMap.put(111L, treeNode_111);treeNodeMap.put(112L, treeNode_112);treeNodeMap.put(121L, treeNode_121);treeNodeMap.put(122L, treeNode_122);treeRich = new TreeRich(treeRoot, treeNodeMap);}@Testpublic void test_tree() {logger.info("决策树组合结构信息:\r\n" + JSON.toJSONString(treeRich));IEngine treeEngineHandle = new TreeEngineHandle();Map<String, String> decisionMatter = new HashMap<>();decisionMatter.put("gender", "man");decisionMatter.put("age", "29");EngineResult result = treeEngineHandle.process(10001L, "Oli09pLkdjh", treeRich, decisionMatter);logger.info("测试结果:{}", JSON.toJSONString(result));}
}
总结
1:从以上的决策树场景来看,组合模式主要解决的是一系列简单逻辑节点或者扩展的复杂逻辑节点在不同结构的组织下,对于外部的调用是仍然可以非常简单的。
2:这部分设计模式保证了开闭原则,无需更改模型结构你就可以提供新的逻辑节点的使用并配合组织出新的关系树。但如果是一些功能差异化非常大的接口进行包装就会变得比较困难,但也不是不能很好的处理,只不过需要做一些适配和特定化的开发。
好了 至此 设计模式之组合模式(营销差异化人群发券,决策树引擎搭建场景) 学习结束了 友友们 点点关注不迷路 老铁们!!!!!
相关文章:

设计模式之组合模式(营销差异化人群发券,决策树引擎搭建场景)
前言: 往往很多大需求都是通过增删改查堆出来的,今天要一个需求if一下,明天加个内容else扩展一下。日积月累需求也就越来越大,扩展和维护的成本也就越来越高。往往大部分研发是不具备产品思维和整体业务需求导向的,总以…...
关于做完 C# 项目的问题总结
1. .Any()方法使用 可以与其他LINQ方法结合使用,以构建更复杂的查询。例如,你可以首先过滤集合,然后检查过滤后的集合是否包含任何元素: List<string> fruits new List<string> { "Apple", "Banana&q…...
CSS响应式布局实现1920屏幕1rem等于100px
代码解析与实现 设置根元素的 font-size 为 5.208333vw 假设你想让根元素的 font-size 基于视口宽度来动态调整。我们可以通过设置 font-size 为 5.208333vw 来让 1rem 相当于视口宽度的 5.208333%。 计算 5.208333vw: 当屏幕宽度为 1920px 时,5.208333vw 等于 5…...
根据当前浏览器版本,下载或更新驱动文件为对应的版本
以前通过ChromeDriverManager().install()的方式自动下载驱动的方式,现在行不通了,访问不通下载网址,会报错:requests.exceptions.ConnectionError: Could not reach host. Are you offline? 所以想着换一个下载地址和方式&…...

【轻量化】YOLOv10 更换骨干网络之 MobileNetv4 | 模块化加法!非 timm 包!
之前咱们在这个文章中讲了timm包的加法,不少同学反馈要模块化的加法,那么这篇就讲解下模块化的加法,值得注意的是,这样改加载不了mobilebnetv4官方开源的权重了~ 论文地址:https://arxiv.org/pdf/2404.10518 代码地址:https://github.com/tensorflow/models/blob/master…...

人体存在感应器设置时间开启感应人存在开灯,失效
环境: 领普人体存在感应器 问题描述: 人体存在感应器设置时间开启感应人存在开灯,失效,设置下午5点,如果有人在5点前一直在这个区域,这个时候到了5点,就触发不了感应自动打开灯光。 解决方案:…...
2024年09月CCF-GESP编程能力等级认证Python编程二级真题解析
本文收录于专栏《Python等级认证CCF-GESP真题解析》,专栏总目录:点这里,订阅后可阅读专栏内所有文章。 一、单选题(每题 2 分,共 30 分) 第 1 题 据有关资料,山东大学于1972年研制成功DJL-1计算机,并于1973年投入运行,其综合性能居当时全国第三位。DJL-1计算机运算控…...
Vuex vs Pinia:新一代Vue状态管理方案对比
引言 随着Vue生态系统的不断发展,状态管理已经成为现代Vue应用程序中不可或缺的一部分。Vuex作为Vue官方的状态管理方案,一直是开发者的首选。然而,随着Pinia的出现,为Vue开发者带来了新的选择。本文将深入对比这两个状态管理方案…...
es查询报错:too_many_buckets_exception
故障排除 es查询报错:too_many_buckets_exception {"error":{"root_cause":[],"type":"search_phase_execution_exception","reason":"","phase":"fetch","grouped":…...

outlook邮箱关闭垃圾邮件——PowerAutomate自动化任务
微软邮箱反垃圾已经很强大了非常敏感,自家的域名的邮件都能给扔到垃圾邮箱里,但还是在本地增加了一层垃圾邮箱功能,然后垃圾邮箱并没有提示,导致错过很多通知,本身并没有提供关闭的功能,但微软有个Microsof…...

机器学习(七)——集成学习(个体与集成、Boosting、Bagging、随机森林RF、结合策略、多样性增强、多样性度量、Python源码)
目录 关于1 个体与集成2 Boosting3 Bagging与随机森林4 结合策略5 多样性X 案例代码X.1 分类任务-Adaboost-SVMX.1.1 源码X.1.2 数据集(鸢尾花数据集)X.1.3 模型效果 X.2 分类任务-随机森林RFX.2.1 源码X.2.2 数据集(鸢尾花数据集)…...

vue跳转传参
path 跳转只能使用 query 传参 ,name 跳转都可以 params :获取来自动态路由的参数 query :获取来自 search 部分的参数...

初识Linux · 共享内存
目录 理解共享内存 Shared memmory code 理解共享内存 前文介绍的管道方式的通信,本文介绍的是进程通信的另外一种方式,即共享内存。但是这种通信方式的特点是只能本地通信,并且不像管道那样有保护机制,这里是没有的。 我们通…...

Illumina测序什么时候会测序到接头序列?
Storage-D: 一个支持实用及个性化 DNA 数据存储的用户友好型平台 iMeta主页:http://www.imeta.science 方法论文 ● 期刊:iMeta(IF 23.7) ● 原文链接DOI: https://doi.org/10.1002/imt2.168 ● 2024年1月21日,中国…...

Element表格show-overflow-tooltip属性
表格默认情况下若内容过多会折行显示,若需要单行显示可以使用show-overflow-tooltip属性,它接受一个Boolean,为true时多余的内容会在 hover 时以 tooltip 的形式显示出来。 <el-table v-loading"loading" :data"list"…...

蓝桥杯竞赛单片机组备赛【经验帖】
本人获奖情况说明 笔者本人曾参加过两次蓝桥杯电子赛,在第十二届蓝桥杯大赛单片机设计与开发组获得省级一等奖和国家级二等奖,在第十五届嵌入式设计开发组获得省级二等奖。如果跟着本帖的流程备赛,只要认真勤奋,拿个省二绝对没问…...
解密复杂系统:理论、模型与案例(3)
第五章:复杂系统的应用案例 复杂系统理论在多个领域中展现出其独特的分析能力和广泛的应用前景。本章将详细探讨复杂系统在生态系统、经济与金融系统、社会网络以及生物系统中的具体应用,通过丰富的案例分析,揭示复杂系统理论在实际问题解决…...

<项目代码>YOLOv8 番茄识别<目标检测>
YOLOv8是一种单阶段(one-stage)检测算法,它将目标检测问题转化为一个回归问题,能够在一次前向传播过程中同时完成目标的分类和定位任务。相较于两阶段检测算法(如Faster R-CNN),YOLOv8具有更高的…...

docker安装到D盘
双击安装docker默认是安装在c盘,并且安装时我们没法选择位置,如果我们要安装在其他盘可以通过命令行安装 1、下载docker https://docs.docker.com/desktop/setup/install/windows-install/ Docker Desktop 可以使用 WSL 和 Hyper-V任意一种架构…...

【Java语言】String类
在C语言中字符串用字符可以表示,可在Java中有单独的类来表示字符串(就是String),现在我来介绍介绍String类。 字符串构造 一般字符串都是直接赋值构造的,像这样: 还可以这样构造: 图更能直观的…...

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式
一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明:假设每台服务器已…...
Python爬虫实战:研究MechanicalSoup库相关技术
一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...

相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了: 这一篇我们开始讲: 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下: 一、场景操作步骤 操作步…...
多场景 OkHttpClient 管理器 - Android 网络通信解决方案
下面是一个完整的 Android 实现,展示如何创建和管理多个 OkHttpClient 实例,分别用于长连接、普通 HTTP 请求和文件下载场景。 <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas…...

YSYX学习记录(八)
C语言,练习0: 先创建一个文件夹,我用的是物理机: 安装build-essential 练习1: 我注释掉了 #include <stdio.h> 出现下面错误 在你的文本编辑器中打开ex1文件,随机修改或删除一部分,之后…...

为什么需要建设工程项目管理?工程项目管理有哪些亮点功能?
在建筑行业,项目管理的重要性不言而喻。随着工程规模的扩大、技术复杂度的提升,传统的管理模式已经难以满足现代工程的需求。过去,许多企业依赖手工记录、口头沟通和分散的信息管理,导致效率低下、成本失控、风险频发。例如&#…...
vue3 字体颜色设置的多种方式
在Vue 3中设置字体颜色可以通过多种方式实现,这取决于你是想在组件内部直接设置,还是在CSS/SCSS/LESS等样式文件中定义。以下是几种常见的方法: 1. 内联样式 你可以直接在模板中使用style绑定来设置字体颜色。 <template><div :s…...

Ascend NPU上适配Step-Audio模型
1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统,支持多语言对话(如 中文,英文,日语),语音情感(如 开心,悲伤)&#x…...

自然语言处理——循环神经网络
自然语言处理——循环神经网络 循环神经网络应用到基于机器学习的自然语言处理任务序列到类别同步的序列到序列模式异步的序列到序列模式 参数学习和长程依赖问题基于门控的循环神经网络门控循环单元(GRU)长短期记忆神经网络(LSTM)…...
浅谈不同二分算法的查找情况
二分算法原理比较简单,但是实际的算法模板却有很多,这一切都源于二分查找问题中的复杂情况和二分算法的边界处理,以下是博主对一些二分算法查找的情况分析。 需要说明的是,以下二分算法都是基于有序序列为升序有序的情况…...