提升团队工程交付能力,从“看见”工程活动和研发模式开始
作者:张裕、雅纯
理想中的研发团队应当具有以下特征:
-
总是工作在最高优先级的事项上
理想的研发团队能够识别并始终集中精力在当前最紧迫和最有价值的任务上。这需要团队具备出色的项目管理能力和决策能力,以便能够正确评估优先级,做出合理的工作分配,并快速适应项目需求的变化。
-
各个角色既能专注于自身的专业工作,又能彼此高效协同
每个团队成员都应当是各自领域的专家,并且全身心投入到他们擅长和负责的工作当中。然而,这并不意味着他们仅限于个体工作。一个理想的研发团队鼓励跨学科合作,通过敏捷的沟通机制和共享的工具,确保信息能够顺畅地在团队成员之间流通。团队中的设计师、工程师、产品经理和其他角色之间应当存在协同工作的文化,这样的多元化合作能够促进创新思维,并最终导致更高质量的产品开发。
-
团队和个体的技术和工程能力能持续改进
理想研发团队不仅在现有技术上精通,而且持续追求技术和专业技能的提升。这意味着个人和团队都应该鼓励创新和实验,并且能做到快速试错、快速反馈。同时,团队应该有制度鼓励个人在工作中尝试新方法和技术,这种文化不仅有助于团队的长期成长,也有助于吸引和保留那些富有好奇心和热情于学习新事物的人才。
一个拥有如上特质的研发团队更有可能成功地完成复杂的项目,创造创新的产品,并在竞争激烈的市场环境中获得成功。
团队工程交付的常见问题
要成为上面所述的优秀研发团队确实需要付出巨大的努力和持续的改进。在追求理想状态的过程中,以下三个问题经常成为阻碍团队达到理想特质的障碍:
1)信息传递失真: 团队内部成员来自不同的专业背景,使用的术语和概念也各不相同。例如,开发说发布一个应用,运维说对一个服务做变更,但他俩说的其实是同一件事情。在日常协作中,需要将这些信息从一个领域转换到另一个领域,并确保信息不丢失、不扭曲。如果处理不当,就会导致团队成员对项目的理解出现偏差,从而影响决策和执行。为了解决这个问题,可以通过建立统一的概念模型、使用共享的术语库、提供跨部门交流培训和使用相同的工具平台等方式来减少信息传递过程中的失真。
2)流程没有连接: 尽管每个研发阶段内部可能已经实现了自动化和高效的运作,但是当一个项目或需求从一个阶段转移到另一个阶段时,往往缺乏流畅的衔接。举个例子,有的企业开发人员和测试人员属于不同的职能团队,开发人员提交代码后,自动会触发代码的构建、静态检查、单元测试等环节,但到了功能测试阶段,开发人员需要手动填写提测单,在提测单里写上代码版本、单测结果、静态检查结果、部署方式等,由测试人员线下确认后,再流转到功能测试阶段。这种阶段间的断层通常需要依赖于团队成员之间的线下沟通和非正式协议,这容易造成流程上的混乱和效率低下。要打破这些障碍,团队可以尝试引入端到端的研发管理工具和流程,确保流程的透明化和自动化,从而形成一个无缝连接的、整体的研发流程。
3)无法识别重点: 当团队同时处理多个项目和需求时,工程活动可能会分散在不同的工具和平台上。这种分散导致团队成员很难追踪整体的进展,也难以判断哪些任务是当前的重点。信息的碎片化使得团队难以集中注意力在最紧迫的需求上。解决这个问题的关键在于建立统一的研发管理系统,按研发任务聚合工程活动,实时展示各个任务的状态和优先级。此外,定期的回顾会议和优先事项的重新评估也是确保团队能够集中精力在最有价值的工作上的重要做法。
总结来说,成为一个优秀的研发团队不仅需要专业技能的不断提升,而且还需要针对信息流通、流程衔接和重点识别等方面的问题进行系统的解决方案设计和实施。通过持续的努力,优秀的团队可以逐步克服这些拦路虎,走向成熟和效能的最高标准。
因此,改进的第一步是要能看见工程活动和研发模式,进而识别其中存在的问题。
统一工程交付的概念模型
为了有效解决信息传递失真、流程不连贯等问题,确保信息的流畅传递和流程的无缝连接是至关重要的。这就要求从根本上统一工程交付的概念模型,使所有参与者——无论是开发人员、测试人员、产品经理还是任何其他相关方——都拥有共同的理解框架。
在解决这些问题的过程中,云效联合产学研各界于 2022 年发布了 BizDevOps白皮书, 该白皮书提出了 BizDevOps 完整的概念模型,通过该模型,可以更清晰地界定和管理研发生命周期中的各个环节。
具体到模型本身,它将业务需求、产品需求、变更请求定义为时标对象,这些时标对象在时间轴上代表了需求的生成和变更的发生。每一个变更请求都与特定的应用相关联,而应用就是变更请求所属的空间或上下文。这样,工程交付的核心概念就被简化为两个主要元素:应用和变更请求。此外,还包括了应用的一些重要附属属性,例如变更内容、环境、部署编排和研发变更流程等。这些属性共同描述了从需求提出到最终部署的完整过程。
通过应用这个核心概念,工程侧能够高效地聚合研发资产和研发流程,形成一个集中的管理点。这有助于优化资源分配,提高研发效率,同时也有助于跟踪和度量研发过程中的关键指标。
另一方面,变更请求作为时标对象,承担了连接不同研发活动和项目协作的关键角色。通过对变更请求的跟踪和管理,团队可以确保所有的活动都围绕着实现具体的业务目标进行,同时使得整个工程交付过程更加透明和可控。
综上所述,这个模型不仅为团队成员之间的沟通提供了共同的语言,还为整个研发周期的管理提供了一套清晰的指南,从而使得各个环节能够紧密协作,确保研发活动能够高效、有序地进行。
定义应用交付的模式
拥有了统一的概念模型后,我们得以实现对研发资产和流程的系统化规范和高效管理。具体来看:
1)基于应用将研发资产和研发流程有效地规范和管理起来: 我们为此构建了一套标准化模板,旨在帮助团队对应用的研发资产和流程进行全面梳理。这个模板涵盖的内容包括但不限于:
a. 应用相关角色及其权限: 定义每个涉及应用开发的角色(如开发人员、测试工程师、产品经理等)以及它们相对于应用的权限,确保权限的分配既满足安全要求又促进工作效率。
b. 应用的代码和制品: 明确代码库管理和制品库的使用,以及不同角色在代码提交、审核、制品生成和存储过程中的职责和权限。
c. 应用的分支模式: 规定了源代码管理中各种分支的使用场景和规范,以及不同分支对应角色的职责,确保代码的版本管理既清晰又高效。
d. 应用端到端的研发流程: 详细描述了从开发任务的启动到产品的最终上线,涉及的所有阶段和流水线,包括每个阶段的具体任务、责任分配、准入和准出标准,以及阶段间的衔接方法。
e. 应用的环境及其与角色的对应关系: 梳理各种环境(如开发环境、测试环境、生产环境)的配置和用途,以及各个环境中不同角色的责任和权限。
2)基于变更请求将产品需求和开发任务端到端地连接起来: 与上面的静态资产和流程管理相比较,这里更侧重于需求到上线这一动态的研发流程。
a. 创建变更请求: 这一流程的第一步通常是将产品需求转化为技术任务,即变更请求,这些变更请求直接属于相应的应用。
b. 指定变更范围: 变更请求的创建过程中,会指定其变更范围,通常指定为某个代码库的特性分支。开发人员在此分支上进行代码提交,触发应用的研发流程。
c. 执行研发流程: 随着研发流程的展开,变更请求会逐渐通过各个阶段,特性分支也可能会被合并到集成分支或发布分支。每个阶段的执行频率可能不同,一般情况下,越接近流程的末端,执行的次数就越少。
d. 完成变更: 当变更请求成功通过最后一个阶段,它就被视为完成。同理,一个产品需求所对应的所有变更请求一旦全部完成,那么这个产品需求也就可以宣布完成或者发布上线。
基于云效平台的落地方法
我们强烈建议在落地工程交付实践之前,先把需求协作实践梳理清楚,关于这一块内容,推荐参考:如何制定科学有效的需求流程规范。
接下来,我们会借助云效平台,按照前面章节的示例,定义应用的交付模式,并按照该交付模式完成一个产品需求交付的完整流程。
4.1 通过应用模板定义应用交付模式
我们通过应用模板来承载团队的工程交付模式,这里我们以前面提到过的基于 feature 的持续交付模式为例。
该交付模式的特点是开发、测试均基于特性分支,集成发布均基于主干分支,属于快速开始,快速集成,快速交付,推崇单个特性的独立开发、独立测试、独立集成于独立交付。
首先,在云效 appstack 上创建一个名为“特性驱动的持续交付模板”的应用模板。
在该模板上开启“变更 + 研发流程”服务。
按照 feature/master 两阶段的研发流程,为这两个阶段分别定义变量组,在变量组中使用不同的 k8s namespace,以及指定不同的副本数。
接下来通过模板来规范应用的部署方式,云效推崇多套环境一套编排模板的实践,差异性的部分通过变量组来定义。
然后,我们规定每个应用都有两套环境,分别为用于 feature 开发验证的“特性验证环境”,和用于集成发布的“生产部署环境”。这两套环境与对应的变量组、部署编排和集群资源(可选)关联。
我们已经确定了应用的环境和部署策略,接下来我们规范应用的研发交付流程。
我们要求应用从开始开发到完成交付,需要经过特性验证和生产部署两个阶段的验证,且只有经过特性验证阶段的 feature,才能进行生产部署。为了做到这一点,我们创建了一个两阶段的研发流程,分别为特性验证阶段和生产部署阶段。
在特性验证阶段,我们定义了一条包含 4 个步骤的流水线,分别为代码检视、构建、部署和测试,且规定分支为自由选择方式(可在流水线配置名称前缀为 feature- 的分支有新的代码提交自动触发)。
在生产部署阶段,我们配置了一条有 5 个步骤的流水线,分别为代码检视、构建、审核、部署和完成变更。同时限制流水线运行分支为 master,且执行时相关 feature 在特性验证阶段的执行结果为成功(云效会自动计算流水线执行时所涉及到的 feature 分支,并判断其前序阶段的执行成功与否)。
至此,我们完成了应用模板的定义,现在,让我们基于该模板来创建一个应用,并完成一个特性的交付。
通过应用模板创建好应用后,还需要设置好应用所关联的代码仓库和相关成员。
4.2 分析产品需求,拆解变更请求
假设我们接到一个产品需求,需要将查询服务接入风控,避免爬虫攻击。为此,我们为 risk-control-srv 拆解了一个变更请求,并关联到该产品需求上。
4.3 在变更分支上提交代码,进行持续验证
由于设置了代码提交至 feature 分支自动触发特性验证阶段的执行,每次在 feature 上 push 代码后,都会自动进行验证并给出反馈。
4.4 通过代码评审合并变更分支,进入生产部署
当代码评审通过并合并入 master 分支后,会自动触发生产部署阶段的执行。
4.5 完成变更,进而完成产品需求
生产部署阶段执行完成后,变更请求会变为已完成状态,同时其对应的产品需求也会自动进入已完成状态。
后记
本文从统一工程交付的概念模型开始,介绍了如何将应用交付的模式显式地定义出来,并通过工具平台落地。但需注意,团队的工程交付实践往往不存在标准解,我们都是在寻求当前场景下的最优解。在具体的场景下,团队的工程交付受到协作机制和技术水平的双重制约,因此需要我们把视角从工程交付本身跳出来,结合协作、技术一起来看,并持续优化和改进,才能找到适合我们自身团队的最佳实践模式。
相关文章:

提升团队工程交付能力,从“看见”工程活动和研发模式开始
作者:张裕、雅纯 理想中的研发团队应当具有以下特征: 总是工作在最高优先级的事项上 理想的研发团队能够识别并始终集中精力在当前最紧迫和最有价值的任务上。这需要团队具备出色的项目管理能力和决策能力,以便能够正确评估优先级࿰…...

前端学习之DOM编程案例:全选反选案例
代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>全选反选</title> </head> <body><input type"checkbox" id"all">全选<ul><li><…...
golang map
1.底层实现 2.如何解决hash冲突 3.扩容机制 4.无序 5.非线程安全 6.不可寻址 runtime/map.go 1.底层实现 底层基于hash表实现,实现有2个结构体hmap,bmap,map由若干个桶存储,每个桶存8个元素,使用链地址解决hash冲突 …...
设计模式:享元模式案例
让我们以游戏开发中的棋类游戏(例如国际象棋)为例来展示享元模式的代码实现。在这个例子中,棋子的类型是内部状态,而棋子的位置是外部状态。 Java 代码示例 import java.util.HashMap; import java.util.Map;// 享元接口 interf…...
pandas(day5)
一. 检测重复值 1.1 检测 data pd.read_csv("./teacher/订单数据.csv")检测行与行之前是否有重复值 data.drop_duplicates()检测 列是否有重复值出现, keep first 从前往后判定 , last是从后往前判定data.drop_duplicates(subset["产…...

如何注册midjourney账号
注册Midjourney账号比较简单,准备好上网工具,进入官网 Midjourney访问地址: https://www.midjourney.com/ 目前没有免费使用额度了,会员最低 10 美元/月,一般建议使用30美元/月的订阅方案。了解如何订阅可以查看订阅…...

探索数据结构:特殊的双向队列
✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:数据结构与算法 贝蒂的主页:Betty’s blog 1. 双向队列的定义 **双向队列(double‑ended queue)**是一种特殊的队列…...
16_I2C库函数
I2C库函数 1.void I2C_DeInit(I2C_TypeDef* I2Cx);2.void I2C_Init(I2C_TypeDef* I2Cx, I2C_InitTypeDef* I2C_InitStruct);3.void I2C_StructInit(I2C_InitTypeDef* I2C_InitStruct);4.void I2C_Cmd(I2C_TypeDef* I2Cx, FunctionalState NewState);5.void I2C_DMACmd(I2C_Type…...
十八、Rust gRPC 多 proto 演示
十八、Rust gRPC 多 proto 演示 网上及各官方资料,基本是一个 proto 文件,而实际项目,大多是有层级结构的多 proto 文件形式,本篇文章 基于此诉求,构建一个使用多 proto 文件的 rust grpc 使用示例。 关于 grpc 的实现…...
【Linux】Linux64位环境下编译32位报错skipping incompatible的解决办法
本文首发于 ❄️慕雪的寒舍 问题 如题,当我尝试在wsl2的ubuntu中使用-m32选项编译32位程序的时候,出现了下面的两种报错 ❯ g -m32 test.cpp -o test1 && ./test1 In file included from test.cpp:1: /usr/include/stdio.h:27:10: fatal error…...
vue指令v-model
<!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8"> <meta name"viewport" content"widthdevice-width, initial-scale1.0"> <title>vue指令v-model</title> </head>…...
CentOS安装MySQL数据库
一、更新yum源 #下载对应repo文件 wget -O CentOS-Base.repo http://mirrors.aliyun.com/repo/Centos-8.repo #清除缓存 yum clean all #生成新缓存 yum makecache #更新 yum update -y 二、安装MySQL #获取源 wget http://repo.mysql.com/mysql80-community-release-el7-3.…...

从B2B转向B2B2C模式:工业品牌史丹利百得的转型历程
图片来源:Twitter 在当今数据驱动的营销环境中,企业努力更好了解客户,并在整个客户旅程中提供个性化体验。史丹利百得(Stanley Black & Decker)是一家领先的工具和工业设备供应商,近年来开始重大转型。…...

Redis群集模式和rsync远程同步
一、Redis群集模式 1.1 概念 1.2 作用 1.2.1 Redis集群的数据分片 1.2.2 Redis集群的主从复制模型 1.3 搭建Redis 群集模式 1.3.1 开启群集功能 1.3.2 启动redis节点 1.3.3 启动集群 1.3.4 测试群集 二、rsync远程同步 2.1 概念 2.2 同步方式 2.3 备份的方式 2.4…...

JAVA—抽象—定义抽象类Converter及其子类WeightConverter
同样,我们由这道题引出抽象类,抽象方法这个概念。 按下面要求定义类Converter及其子类WeightConverter 定义抽象类:Converter: 定义一个抽象类Converter,表示换算器,其定义的如下: 一个私有…...

面对复杂多变的网络攻击,企业应如何守护网络安全
企业上云,即越来越多的企业把业务和数据,迁移到云端。随着云计算、大数据、物联网、人工智能等技术的发展,用户、应用程序和数据无处不在,企业之间的业务边界逐渐被打破,网络攻击愈演愈烈,手段更为多。 当前…...

计算机网络练习-计算机网络概述与性能指标
计算机网络概述 ----------------------------------------------------------------------------------------------------------------------------- 1. 计算机网络最据本的功能的是( )。 1,差错控制 Ⅱ.路由选择 Ⅲ,分布式处理 IV.传输控制 …...

vite vue3 ts import.meta在vscode中报错
问题描述:开发使用的框架为vitevue3ts,在开发过程中莫名其妙报仅当“--module”选项为“es2020”、“esnext”或“系统”时才允许使用“import.meta”元属性 问题解决: 通过更改tsconfig.json的module为esnext,es2022等࿰…...
Java synchronized(详细)
synchronized 一,介绍 在Java中,synchronized关键字用于解决多线程并发访问共享资源时可能出现的线程安全问题。当多个线程同时访问共享资源时,如果没有合适的同步机制,可能会导致以下问题: 竞态条件(…...

算法设计与分析实验报告python实现(排序算法、三壶谜题、交替放置的碟子、带锁的门)
一、 实验目的 1.加深学生对算法设计方法的基本思想、基本步骤、基本方法的理解与掌握; 2.提高学生利用课堂所学知识解决实际问题的能力; 3.提高学生综合应用所学知识解决实际问题的能力。 二、实验任务 1、排序算法…...

装饰模式(Decorator Pattern)重构java邮件发奖系统实战
前言 现在我们有个如下的需求,设计一个邮件发奖的小系统, 需求 1.数据验证 → 2. 敏感信息加密 → 3. 日志记录 → 4. 实际发送邮件 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其…...

Python:操作 Excel 折叠
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 Python 操作 Excel 系列 读取单元格数据按行写入设置行高和列宽自动调整行高和列宽水平…...

MongoDB学习和应用(高效的非关系型数据库)
一丶 MongoDB简介 对于社交类软件的功能,我们需要对它的功能特点进行分析: 数据量会随着用户数增大而增大读多写少价值较低非好友看不到其动态信息地理位置的查询… 针对以上特点进行分析各大存储工具: mysql:关系型数据库&am…...
渲染学进阶内容——模型
最近在写模组的时候发现渲染器里面离不开模型的定义,在渲染的第二篇文章中简单的讲解了一下关于模型部分的内容,其实不管是方块还是方块实体,都离不开模型的内容 🧱 一、CubeListBuilder 功能解析 CubeListBuilder 是 Minecraft Java 版模型系统的核心构建器,用于动态创…...
linux 错误码总结
1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...

srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...
Spring Boot+Neo4j知识图谱实战:3步搭建智能关系网络!
一、引言 在数据驱动的背景下,知识图谱凭借其高效的信息组织能力,正逐步成为各行业应用的关键技术。本文聚焦 Spring Boot与Neo4j图数据库的技术结合,探讨知识图谱开发的实现细节,帮助读者掌握该技术栈在实际项目中的落地方法。 …...
三体问题详解
从物理学角度,三体问题之所以不稳定,是因为三个天体在万有引力作用下相互作用,形成一个非线性耦合系统。我们可以从牛顿经典力学出发,列出具体的运动方程,并说明为何这个系统本质上是混沌的,无法得到一般解…...

Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)
参考官方文档:https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java(供 Kotlin 使用) 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...

html css js网页制作成品——HTML+CSS榴莲商城网页设计(4页)附源码
目录 一、👨🎓网站题目 二、✍️网站描述 三、📚网站介绍 四、🌐网站效果 五、🪓 代码实现 🧱HTML 六、🥇 如何让学习不再盲目 七、🎁更多干货 一、👨…...