得物技术 前天 20:27
DPP推荐引擎架构升级演进之路|得物技术
index_new5.html
../../../zaker_core/zaker_tpl_static/wap/tpl_guoji1.html

 

本文深入探讨了DPP推荐引擎的演进历程,从最初的固定编排到灵活编排,再到图化DAG编排,详细介绍了每个阶段的特点和优势。文章重点阐述了图化DAG编排的设计理念、核心组件以及在实际应用中的效果,旨在通过技术创新不断提升策略迭代效率,优化系统性能,并最终实现全链路的图化。文章还分享了得物在这一领域的实践经验,为读者提供了宝贵的参考。

🚀 DPP引擎的演进经历了三个关键阶段:固定编排(DPP-Engine)、灵活编排(BizEngine)和图化DAG编排,每个阶段都旨在提升策略迭代效率。

⚙️ BizEngine通过引入组件和编排流程,提高了推荐系统的灵活性,但同时也暴露出组件粒度问题和线程调度冲突,限制了其进一步发展。

💡 图化DAG编排通过将业务逻辑抽象为DAG图,实现了算子级别的复用和灵活定制,降低了开发成本,提升了性能,并为算法开发提供了统一接口。

🧩 图化框架的核心在于算子、图和子图的设计。算子定义了标准接口和协议,图和子图则通过配置文件描述业务逻辑,实验配置决定了配置的生效方式。

📈 图化DAG编排在广告和社区等场景中取得了显著的性能提升,新的开发模式促使策略同学专注于算子级别的实现,并简化了场景编排迭代流程。

原创 在东 2025-04-07 18:31 上海

DPP依赖于算法平台的引擎服务,提供“开箱即用”的召回,粗排,精排服务。采用“热加载技术”解决算法平台的工程和算法同学策略迭代效率问题,支持策略随时发布,让他们可以专注于业务逻辑,即可拥有稳定的推荐在线服务。

目录

一、DPP整体架构

    1. 平台特性

二、DPP引擎演进

    1. 固定编排 - DPP-Engine

    2. 灵活编排 - BizEngine

    3. 图化核心设计和协议

三、总结

DPP整体架构

DPP依赖于算法平台的引擎服务(FeatureServer,召回引擎, 精排打分),提供“开箱即用”的召回,粗排,精排服务。采用“热加载技术”解决算法平台的工程和算法同学策略迭代效率问题,支持策略随时发布,让他们可以专注于业务逻辑,即可拥有稳定的推荐在线服务。


图1.0 DPP服务整体架构


平台特性

    快速迭代:通过系统解耦,实现算法、策略的快速迭代。

    效果分析自动化:打通数据平台,BI数据分析标准化。

    灵活实验:通过分层实验平台,支持多层多实验的灵活配置。

    诊断方便:落地各子流程中间结果,支持算法、策略的细化分析;提供方便的监控告警,运维,时光机等问题排查工具。


DPP引擎演进

DPP编排引擎的迭代分为了3个阶段:固定编排,灵活编排,图化DAG编排;均是在策略迭代过程中,围绕着“迭代效率”提升的不断进化。下面分别介绍下各阶段引擎产生的背景及其方案。


固定编排 - DPP-Engine

推荐业务一般都可以抽象为“召回->融合->粗排->精排->干预”等固定的几个阶段,每个阶段通常是有不同的算法或工程同学进行开发和维护,为了提升迭代效率,通过对推荐流程的抽象,将各阶段的逻辑抽象为“组件"+"配置”,整体的流程同样是一个配置,统一由“编排引擎”进行调度,同时提供统一的埋点/日志等。让工程或算法同学可以关注在自己的业务模块和对应的逻辑,而框架侧也可以做统一的优化和升级。


DPP-Engine就是在此基础上,将业务策略抽象为“初始层->召回层->融合层->粗排层->精排层->干预层”这6层, 有DPP负责串行调度这6层,每一层有若干个组件组成,各层将结果进行合并后传递到下一层(也就是List)。


图1.2-1 DPP-Engine层编排


通过分层,DPP-Engine较好的支持了业务的快速迭代,业务“各层”的开发同学可以独立迭代。但是随着场景的增多,对“灵活”编排有了更多的需求,比如不固定6层,层内可有自己的"编排"等。


其次对于DPP平台同学来说,DPP-Engine嵌入在DPP系统内, 不利于引擎的迭代和维护。


灵活编排 - BizEngine

BizEngine根据策略同学提供的组件及其编排流程,负责执行和调度,包括组件间的并发。它在推荐系统链路中的位置如下图:


图1.3-1 DPP系统(BizEngine)


目前在BizEngine看来,“组件”是策略开发的最小粒度,策略同学在DPP-后台中可以在场景维度划分桶(小流量桶, 分层桶),在桶可以配置不同的层编排,默认为6层:INIT层->召回层->融合层->粗排层->精排层->干预层。分别在层内可以配置不同的组件。一次请求中,BizEngine负责按层进行调度(层与层之间为串行调度),层内的组件根据组件间的依赖进行串行或者并发调度。


图1.3-2 编排管理及其配置协议


用户请求到DPP后, 会通过AB分流得到该请求(用户)命中的所有实验(包括桶,层,实验),DPP解析命中配置后,可以构建出BizEngine需要的入参-编排配置(桶配置+实验配置+组件配置),它会根据层及组件的配置构建出执行的层Stages,按组件维度提交到各线程池进行同步或异步的调度,流程可参考下图:


图1.3-3 BizEngine的组件调度和执行


从上图可以看到我们是按层进行串行调度的,“分层”是按推荐的业务策略逻辑来分的,符合工程算法同学的分工和职责,特别是算法同学通常有各自负责的领域(召回模型,粗排模型,精排模型,干预),按层划分和进行实验可以有效提高迭代效率,做到相互之间不影响。“组件”则是BizEngine层内调度的单元,但是目前组件的粒度可大可小,比如社区的部分场景,他们在组件内拆分了更细粒度的Steps,并且独立于组件进行调度(依赖DPP场景线程池或自定义线程池),因此策略代码即负责了策略的逻辑, 还需要负责策略逻辑单元(Step)的调度。由此可以看出BizEngine未来的可进一步发展的方向:

    按层进行串行调度,即便层与层组件之间为串行,也需要按层调度,存在一定开销。

    BizEngine的线程调度和策略内自定义调度的冲突,线程池资源难于实现高效利用。

    “组件粒度”问题:目前看策略同学实现的组件对BizEngine来说是“逻辑黑盒”,里面可能是CPU,也可能是IO,也可能是一个发起并发任务的模块,可能涉及自定义的线程池资源。

    随着业务不断迭代, 策略组件的迁移和重构成本逐渐上升;缺少“组件”/“代码”共享及发现的机制,不利于我们通过“组件复用”的方式去提升迭代效率。


图化DAG - DagEngine

为什么需要做图化?

那为什么要去做“图化”/“DAG”呢?其实要真正要回答的是:  如何应对上面看到的挑战?如何解决BizEngine目前发展碰到的问题?


从业界搜推领域可以看到不约而同地在推进“图化”/“DAG”。 从TensorFlow广泛采用之后,我们已经习惯把计算和数据通过采用算子(Operation)和数据(Tensor)的方式来表达,可以很好的表达搜索推荐的“召回/融合/粗排/精排/过滤”等逻辑,图化使得大家可以使用一套“模型”语言去描述业务逻辑。DAG引擎也可以在不同的系统有具体不同的实现,处理业务定制支持或者性能优化等。


通过图(DAG)来描述我们的业务逻辑,也带来这些好处:为算法的开发提供统一的接口,采用算子级别的复用,减少相似算子的重复开发;通过图化的架构,达到流程的灵活定制;算子执行的并行化和异步化可降低RT,提升性能。



图化架构

图化是要将业务逻辑抽象为一个DAG图,图的节点是算子,边是数据流。不同的算子构成子图,用于逻辑高一层的封装,子图的输出可以被其他子图或者算子引用。图化后,策略同学的开发任务变成了开发算子,抽象业务领的数据模型。不用再关心“并行化异步化”逻辑,交由DAG引擎进行调度。“算子”要求我们以较小粒度支持,通过数据实现节点的依赖。


图化定义了新的业务编排框架,对策略同学来说是“新的开发模式”,可分为3个部分:一个是我们会定义算子/图/子图的标准接口和协议,策略同学实现这些接口,构建业务的逻辑图;二是DAG引擎,负责逻辑图的解析,算子的调度,保证性能和稳定性;三是产品化,DAG Debug助手支持算子/图/子图的开发调试,后台侧提供算子/子图/图的可视化管理。整体架构参考下图:


图4.0.0 - DPP图化框架


图4.0.1 - DagEngine


图化核心设计和协议

1.算子

    算子接口定义Processor<O>

    public interface Processor<O> {    /**     * 执行逻辑     *     * @param computeContext 执行上下文信息     * @return 返回执行结果     */    DataFrame<O> run(ComputeContext computeContext, DataFrame... inputs);}


      算子注解@DagProcessor

    通过注解可对算子进行描述和提供运行时信息:

      @Documented@Retention(RetentionPolicy.RUNTIME)@Target({ElementType.TYPE})public @interface DagProcessor {    /**     * 标记IO/CPU, 影响DagEngine的调度     * @return     */    String type() default "IO";    /**     * 算子描述     *     * @return String     */    String desc() default "";    /**     * 用于标识该算子会输出的一些中间值, 可用于做运行时的依赖校验     * 可理解为是算子OP的side effects     */    String sideValues() default "";}


        依赖配置@ConfigAnno)

      算子通过注解(@ConfigAnno) 一是声明算子需要的配置(通过DPP-后台实验配置进行配置), 二是运行时DAG引擎会对注解的值进行注入.


        依赖数据@DependsDataAnno

      算子节点上游的数据,通过接口参数也会透传过来(DataFrame数组),算子内可以通过dataFame.getName()获取数据的唯一标识(请求session内唯一)。


      算子的返回作为该算子的输出数据,通过name可以获取, 比如 @DependsDataAnno(name = "某一路的输出",desc = "recall1")。


      写策略逻辑过程中的中间变量是我们必不可少的,算子可以通过注解@DagProcessor#sideValues声明会输出那些数据(names),通过name 可以获取。


      比如依赖了同一个算子(多个实例),它的输出name是一样的,下游获取需要通过这个优先级决定。


      Note:@DagProcessor#sideValues 可能作为必须的,只有sideValues声明了的数据,才可以被依赖算子引用,这有助于我们管理和防止依赖不存在的数据。


      Note:算子获取sideValue时有多相同name的数据时,通过配置指定算子优先级。


      2.图/子图

        图/子图/配置文件

      图分为图和子图,一个场景可以有多个图,可按垂直桶制定不同的图;子图定位为业务逻辑模版,可以将若干个独立算子组装为具有特定业务含义的“子图”,子图和算子一样可在场景大“图”中进行配置,即运行时可有多个“实例”,实现逻辑的复用和配置化。


      图或子图通过“配置文件”文件来描述,考虑到可读性和是否支持注释等特性,确定选用yaml来定义。


        协议

      子图

        ## 子图(定位为逻辑模版, 包含: 若干个算子及其依赖关系, 子图的配置及其默认值## Note: 子图的配置实际为算子的配置, 在算子中引用name: 'Recall子图1' ## 场景全局唯一type'subgraph' ## 标记图为"子图"configs: ## 子图包含配置项( 指定默认值 )  - name: 'configKey1' ##     value: '默认值Value, 可为string, json等, xx'  # - 其他配置及其默认值  # ...nodes: ## 子图包含的所有算子, 通过dpends指定依赖.  ## 比如一路召回  - name: 'fistRecallOp1'    op: 'com.dag.demo.recrecall.FirstRecallOP'    depends: []    # 指定子图中该算子的默认值    configs:    - name: 'configKey1'      value: 'fistRecallOp1s value'  - name: 'otherRecall1'    op: 'com.dag.demo.recrecall.OtherRecallOP'    depends: ['fistRecallOp1']


          ## 图(场景逻辑描述, 包含若干个算子或子图, 及其他们的依赖关系, 图的配置及其默认值(Note: 图的配置实际为算子的配置, 在算子中引用)name: '场景图Name' ## 场景全局唯一type'graph'configs: ## 图包含配置项( 指定默认值 )  - name: 'configKey1'    value: '默认值Value, 可为string, json等'  # - 其他配置及其默认值  # ...nodes: ## 图包含的所有算子或子图, 通过dpends指定依赖.  ## 比如一路召回  - name: 'fistRecallOp1'    op: 'com.dag.demo.recrecall.FirstRecallOP'    depends: []  - name: 'otherRecall1'    op: 'com.dag.demo.recrecall.OtherRecallOP'    depends: ['fistRecallOp1']  ## 子图1( 为`Recall子图1`的实例 )  - name: 'someRecallComplex1'    op: '$Recall子图1' ## 依赖该子图    configs: ## 子图包含配置项( 指定默认值 )      - name: 'configKey1'         value: 'fistRecallOp1s value'        ## 覆盖这两个算子的默认值        targets: ['recallGroup1''dssmRandomBatchRecall']      ## todo 修改op的配置      ##     depends: ['fistRecallOp1']  ## 子图2( 为`Recall子图1`的实例 )  - name: 'someRecallComplex1'    op: '$Recall子图1' ## 依赖该子图    depends: ['fistRecallOp1']


          3.算子配置如何获取? 如何配置?

          图通过算子(子图)+数据依赖的DAG描述了业务的逻辑关系,配置的作用就是影响逻辑如何生效。这些配置通过“实验/AB”来决定,不同的实验就是对图或算子的不同配置。


            默认值

          配置的默认值通过两种方式指定:1/ 算子变量的默认值(代码方式);2/ 图或者子图的Confgis#key#defaultValue


            运行时的值

          算子某个配置在运行时的值,是通过该次请求命中的所有实验进行配置融合和覆盖后得到的。


            如何配置?

          实验配置中:

          需要考虑配置key在子图和算子中的name作为前缀,规则为<subGraph'sName>.<op'sName>.<key'sName>,若算子不在子图中(即, 直接配置在主图中),那么配置为_.<op'sName>.<key'sName>。


          算子代码中:

          通过注解 @ConfigAnno(key = "key'sName")来获取对的key'sName的值. 运行时DAG引擎负责识别<subGraph'sName> 和<op'sName>

               

          配置支持json和dto对象绑定,DAG运行时实现缓存和校验指定Json配置和类的映射,@ConfigAnno(key = "somepojo.value",isJson = true,clazz = SomePojo.class),DAG引擎负责反序列化。


          图化相关特性/结果

            DPP图化落地广告/社区等场景。



            图桶推全SOP流程: 通过引入"分支"概念,图桶推全变为合入Master,待推全各桶由各Owner自行合并Master。支持一分支绑定多桶。简化了场景编排迭代流程。

            图编辑可视化: 支持算子及其依赖的表单化修改,提升修改效率和易用性。


          总结

          DPP编排引擎经历了固定编排,灵活编排到图化DAG编排三个阶段,持续提升策略迭代效率。



          图化DAG编排在我们落地的一些场景中显著提升了性能,同时新的开发模式要求策略同学关注算子级别的实现,减少对调度逻辑的关注。在产品侧DPP-后台提供了产品化工具支持本地调试和可视化管理。


          未来我们可以进一步探索图化DAG编排在更多业务场景中的应用,尤其是需要高性能和灵活定制的场景。其次加强算子复用机制和标准化建设,降低组件迁移与重构成本, 持续优化DagEngine的高性能特性,如DataFrame数据结构的使用,以进一步提升系统性能。 并且随着引擎及机器学习平台图化的推进,我们有可能也去端到端链路上实现“全图化”。用一张图描述一个业务的策略逻辑。


          往期回顾


          1.Cursor 在前端需求开发工作流中的应用|得物技术

          2.得物 iOS 启动优化之 Building Closure

          3.分布式数据一致性场景与方案处理分析|得物技术

          4.从对话到自主行动:AI应用如何从 Chat 进化为 Agent?开源项目源码深度揭秘|得物技术

          5.得物技术部算法项目管理实践分享

          文 / 在东


          关注得物技术,每周一、三更新技术干货

          要是觉得文章对你有帮助的话,欢迎评论转发点赞~

          未经得物技术许可严禁转载,否则依法追究法律责任。

          扫码添加小助手微信

          如有任何疑问,或想要了解更多技术资讯,请添加小助手微信:


          阅读原文

          跳转微信打开

          Fish AI Reader

          Fish AI Reader

          AI辅助创作,多种专业模板,深度分析,高质量内容生成。从观点提取到深度思考,FishAI为您提供全方位的创作支持。新版本引入自定义参数,让您的创作更加个性化和精准。

          FishAI

          FishAI

          鱼阅,AI 时代的下一个智能信息助手,助你摆脱信息焦虑

          联系邮箱 441953276@qq.com

          相关标签

          DPP 推荐引擎 图化DAG 编排 算法
          相关文章