反思一次效能提升

前天与一个大佬交流。想起自己在6年多前在团队里做的一次小小的效能提升。
改进前
在同一个产品团队,同时有前端工程师和后端工程师。他们经常需要共同协作完成features。
前端是一个传统的多页应用。前端渲染是由后端的velocity模板引擎实现的。
打包后,最终执行就是一个jar包:![[hellojar.png]] vm文件后缀名是velocity模板文件。它们内容大概是这样的:
<html><body>Hello $customer.Name!<table>#foreach( $mud in $mudsOnSpecial )#if ( $customer.hasPurchased($mud) )<tr><td>$flogger.getPromo( $mud )</td></tr>#end#end</table></body>
</html> 其中一些非html代码就是Velocity Template Language。
默认情况下,一个前端工程师是不懂这门模板语言的。而且,这种模板语言对浏览器不友好,不像Thymeleaf。同时,按前端的开发习惯,他们是不可能先启动你一个Java进程后,再对前端页面进行调试。
总的来说,这样的技术栈是不利于前端本地开发的。
所以,在改进前,前后端的协作的方式是:
1. 前后端同时进行开发;
2. 后端负责在Java工程中实现后端的逻辑;
3. 前端负责在另一个独立的前端工程中开发HTML和CSS;
4. 前端完成页面的开发后,他会把前端页面html文件名和页面的逻辑告诉后端;
5. 后端再把html页面里内容与Java工程中的vm进行对比,然后将不同的部分转换到Java工程的vm模板页面中。
这样的协作方式下,经常出现以下问题:
1. 在将html转换到vm模板的过程中,后端有时会转换漏内容;
2. 转换后,前端调试前端问题会很麻烦。前端需要占用一个后端和他一起调试。因为前端不懂如何启动复杂的后端工程;
3. 命名上经常出现不一致,进而导致前端bug。对于同一个概念的情况下,前端命名叫item,后端叫project;
效能提升措施与效果
经分析,以上问题均由“技术之间的转换”导致,即人工的将html页面从一个前端工程转换到一个后端工程的vm模板语言。
所以,解决以上问题的思路就是:最好能消除html和vm的转换,又或者能减少这个转换过程的失误。
思路有了以后,解决方案有以下几个:
1. 更换前端技术栈,不再使用Java+vm模板;
2. 让前端学习vm模板语言,转换过程由前端完成;
3. 优化后端开发在本地启动的流程,方便前端也能在本地启动;
4. 减小测试环境的部署难度。让任何人都可以部署,方便前端进行调试。
方案1短时间无法做到,而且改动巨大,收益也未知。所以,最终是同时做了2、3、4。
虽然当时没有进行度量,但实际效果就是前文提到的所有问题都得到了减轻。
原因就是方案2、3、4,减少了前端需要向后端传递的信息。很多事情,前端一个人就可以解决了。
关于方案2、3,可能有人会好奇:前端去学习一门新的模板语言,成本大吗?前端在本地启动一个后端的工程去调试前端代码,成本高吗?
首先,vm模板并不是一门难的语言,只要有类C语言的基础(比如Java、JavaScript、C#等),都不难学。无非就是if-else判断、变量定义、循环等。
其次,在后端优化本地开发环境后,前端启动一个Java工程并不是一件难事。
反思
整个事件下来,本质上是前端本地开发习惯与现有技术栈的冲突。这次效能的提升需要前端开发做一些工作习惯上妥协。
而这只是表面上的问题,我们需要去思考更深层的问题:
1. 团队成员在改进前为什么一直没有考虑如何改进呢?又或者不知道该如何改进?
2. 对于这次效能改进,感观上是提升了,但是该如何度量呢?
思考问题1是为了让团队的所有的人有意识和思路地提升效能。思考问题2是为了让实践有数据支撑。
答案是什么呢?留给读者朋友思考。
Meta的降本增效三大招:
Meta降本增效大招之:自动删除数据
Meta降本增效大招之:自动清理死代码
Meta降本增效大招之:弃用产品
相关文章:
反思一次效能提升
前天与一个大佬交流。想起自己在6年多前在团队里做的一次小小的效能提升。 改进前 在同一个产品团队,同时有前端工程师和后端工程师。他们经常需要共同协作完成features。 前端是一个传统的多页应用。前端渲染是由后端的velocity模板引擎实现的。 打包后,…...
ElasticSearch之cat indices API
命令样例如下: curl -X GET "https://localhost:9200/_cat/indices?vtrue&pretty" --cacert $ES_HOME/config/certs/http_ca.crt -u "elastic:ohCxPHQBEs5*lo7F9"执行结果输出如下: health status index uuid …...
Composer update 跳过指定依赖
在使用Compose进PHP 依赖管理只时,有时候我们可能希望忽略版本批配,即使依赖项的景新版本已经发布,也然续使用当前的乐本。这种情况下,我们可以使用Composer的 --ignore-platform-reqs 选项来实现 可以使用--ignore-platform-req…...
@RequestMapping详解:请求映射规则
目录 请求-相应模式: 设置请求映射规则RequestMapping POST 请求: GET 请求 请求-相应模式: 前端作为客户端向后端发送请求(请求可以分为请求头和请求体两部分,请求头包含了一些元数据信息,如请求方式、…...
C#中密封类和密封方法
目录 一、定义与特性 1.何时使用密封类 2.定义 3.特性 二、示例 如果所有的类都可以被继承,很容易导致类的层次结构变得十分复杂。使对类的理解和使用变得十分困难。为了避免滥用继承,C#中提出了密封类的概念。 一、定义与特性 密封类可以用来限制…...
Pytorch中的Net.train()和 Net.eval()函数讲解
目录 前言1. Net.train()2. Net.eval()3. 总结 前言 这两个方法通常用于训练和测试阶段 1. Net.train() 该代码用在训练模式中 主要作用: 模型启用了训练时特定的功能(Batch Normalization 和 Dropout)。 在这种模式下,模型会根…...
氪了几百亿,字节游戏停止了“跳动”
目录 一、氪了几百亿,字节游戏停止了“跳动” 二微软推出跨平台框架 ML.NET 3.0 版:强化深度学习、加强AI效率 一、氪了几百亿,字节游戏停止了“跳动” 朝夕光年,扑了 11月26日,脉脉社区的一个截图内容引起大众热议…...
进入docker容器
学习如何进入一个正在运行的容器的内部,要求学习者参照示例,进入一个名为container2的容器内部,并在容器内部创建一个1.txt文件。 相关知识 使容器在后台运行 因为本关要使用docker run -d命令,所以在本关的开始,将…...
C陷阱与缺陷——第5章库函数
1. 返回整数的getchar函数 #include <stdio.h>main() {char c;while((c getchar()) ! EOF){putchar(c);} } 上述函数是错误的,原因在于程序中的变量c被声明为char类型,而不是int类型,这意味着c无法容下所有可能的字符,特…...
【C++上层应用】6. 信号 / 中断
文章目录 【 1. signal 函数 】【 2. raise函数 】 信号是由操作系统传给进程的 中断,会提早终止一个程序。在 UNIX、LINUX、Mac OS X 或 Windows 系统上,可以通过按 CtrlC 产生中断。有些信号不能被程序捕获,但是下表所列信号可以在程序中捕…...
树与二叉树堆:堆的意义
目录 堆的意义: 第一是堆的排序,第二是堆的top k 排行问题 堆的 top k 排行问题: 面对大量数据的top k 问题: 堆排序的实现:——以升序为例 方法一 交换首尾: 建立大堆: 根结点尾结点的…...
什么时候适合做ui自动化测试?什么时候做接口自动化测试
UI自动化测试和接口自动化测试都是软件测试中非常重要的部分,它们各自有适合的应用场景。 适合做UI自动化测试的场景包括: 用户界面(UI)变化频繁的应用程序。需要测试用户交互和流程的应用程序。需要验证页面布局、样式和交互的…...
[ABC261E] Many Operations(dp,位运算,打表)
[ABC261E] Many Operations - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) Problem Statement We have a variable X and N kinds of operations that change the value of X. Operation i is represented as a pair of integers (Ti,Ai), and is the following operati…...
一、爬虫-爬取豆瓣电影案例
1、环境配置 你需要一个pycharm和requests第三方库,在安装完成之后即可继续浏览。 2、操作流程 (1)打开豆瓣电影网站,点击排行榜,点击喜剧,检查 (2)可以看到鼠标每次下移࿰…...
4G5G防爆执法记录仪、防爆智能安全帽赋能智慧燃气,可视化巡检巡线,安全生产管控
随着燃气使用的普及,燃气安全问题日益突出。传统应急安全问题处理方式暴露出以下问题: 应急预案不完善:目前一些燃气企业的应急预案存在实用性不高、流程不清晰等问题,导致在紧急情况下难以迅速启动和有效执行。 部门协同不流畅…...
武汉数字孪生赋能工业制造,加速推进制造业数字化转型
随着数字孪生技术的不断推进,互联网、物联网、智能传感技术开始应用到数控机床的远程服务,状态监控,故障诊断,维护管理等方面。武汉数字孪生是在虚拟空间中创建物理对象的高保真虚拟模型,以模拟其在现实世界中的行为提…...
安卓密码框、EditText
目录 1. 基础使用 2. 密码的展示与隐藏 (1) 使用setTransformationMethod方法 (2) 使用setInputType方法 3. imeOptions属性 4. 单行设置 在安卓中使用密码框普遍采用EditText设置inputType"textPassword"的方式。 1. 基础使用 <EditTextandroid:id"…...
ROS命令行工具
1、roscore 在使用ROS之前,首先要启动roscore进程。当我们在终端中运行这个命令时,系统就会启动ROS Master、参数服务器和日志节点。在这之后,就可以运行任何其他的ROS程序/节点了。所以可以在一个终端窗口运行roscore指令&#…...
深入浅出 Golang 中的直接依赖和间接依赖管理
目录 引言 直接依赖 间接依赖 为什么需要间接依赖? 如何管理间接依赖? 小结 引言 Golang 中的依赖管理是使用 go mod 进行管理的。go mod 是 Golang 官方推出的依赖管理工具,可以帮助开发者管理项目的依赖关系,确保项目代码…...
深入Python元编程:了解声明与初始化定制元类
更多资料获取 📚 个人网站:ipengtao.com 简介 在Python中,元编程是指在运行时创建或定制类的编程。元类是Python中最强大的元编程工具之一,允许您控制类的创建过程。元类是类的类,它控制类的实例化,允许您…...
别再只跑Demo了!手把手教你用YOLOv5/v8训练自己的钢材缺陷数据集并部署成Web服务
从零构建工业级钢材缺陷检测系统:YOLOv5/v8实战全流程指南 在工业质检领域,深度学习技术正在掀起一场革命。想象一下,当传统质检员需要花费数小时仔细检查钢材表面的每一寸区域时,一个训练有素的AI系统可以在几毫秒内完成同样的工…...
Qtile配置终极指南:10个Python配置文件编写技巧
Qtile配置终极指南:10个Python配置文件编写技巧 【免费下载链接】qtile :cookie: A full-featured, hackable tiling window manager written and configured in Python (X11 Wayland) 项目地址: https://gitcode.com/gh_mirrors/qt/qtile Qtile是一款功能全…...
OpenClaw多模态实践:千问3.5-27B图片理解+文件整理自动化
OpenClaw多模态实践:千问3.5-27B图片理解文件整理自动化 1. 为什么需要自动化图片管理 上周整理项目资料时,我发现桌面上散落着237张截图——有会议纪要片段、代码报错提示、参考文档关键页,甚至还有随手截的灵感草图。手动分类这些文件花了…...
2026年西安市莲湖区Geo搜索优化排名,专业企业究竟谁能拔得头筹?
在数字化浪潮席卷的今天,Geo搜索优化(地理搜索优化)对于企业的重要性不言而喻。尤其在西安市莲湖区,企业们对于提升自身在Geo搜索中的排名需求愈发迫切。究竟哪家专业企业能够在2026年的竞争中脱颖而出,成为Geo搜索优化…...
CMPS12磁力计寄存器级驱动与KRAI架构嵌入式实践
CMPS_KRAInew:基于KRAI架构的CMPS12磁力计寄存器级驱动解析与嵌入式集成实践1. 项目概述CMPS_KRAInew 是一个面向嵌入式平台、专为 CMPS12 数字罗盘模块设计的轻量级底层驱动库,其核心定位并非通用 HAL 封装,而是聚焦于 KRAI(Kern…...
大模型训练实战:分布式训练、显存优化与知识蒸馏全解析!
全景路线图: 我们将按模块逐步展开,每个模块都是最终搭建完整平台的一块拼图:之前的章节参考我之前写的文章;G. 分布式训练篇:大模型训练的工程实践 – 学习在多卡多机环境下训练大模型的方法,包括数据并行…...
基于粒子群算法的IEEE33节点配电网无功优化及其结果分析
基于粒子群算法的配电网无功优化 基于IEEE33节点配电网,以无功补偿器的接入位置和容量作为优化变量,以牛拉法进行潮流计算,以配电网网损最小为优化目标,通过优化求解,得到最佳接入位置和容量,优化结果如下所…...
嵌入式蜂鸣器非阻塞管理库BuzzerManager深度解析
1. BuzzerManager 库深度解析:面向嵌入式系统的多路无阻塞蜂鸣器管理方案在嵌入式系统开发中,声音反馈是人机交互最基础、最可靠的物理通道之一。从工业设备的状态提示、医疗仪器的报警响应,到消费电子的按键确认、玩具的音效反馈,…...
OpenSSH安全升级指南:如何快速禁用CBC模式并切换到CTR加密(附最新配置命令)
OpenSSH安全加固实战:从漏洞检测到加密算法升级全流程 最近在给某金融客户做安全审计时,发现他们的生产服务器还在使用OpenSSH的CBC模式加密。这让我想起十年前那个著名的CVE-2008-5161漏洞——攻击者可以利用CBC模式的弱点,从SSH会话中恢复出…...
RPA流程进阶:在Uibot中集成Python插件实现复杂数据处理
1. 为什么要在Uibot中集成Python插件? 很多刚开始接触RPA的朋友可能会有疑问:Uibot本身已经提供了丰富的自动化命令,为什么还要折腾Python插件?这个问题我在三年前第一次用Uibot处理Excel报表时就深有体会。当时需要合并20多个分公…...
