当前位置: 首页 > article >正文

美团面试:为什么要用分布式缓存?本地缓存呢?多级缓存一致性如何保证?创

从 UI 工程师到 AI 应用架构者13 年前我的工作是让按钮在 IE6 上对齐13 年后我用 fetch-event-source 订阅大模型的“思维流”用 OCR 解锁图片中的文字——前端正在成为 AI 产品的第一道体验防线。最近我基于 Vue 3 Vite TypeScript Ant Design Vue 开发了一款企业级 AI 多语翻译平台。它支持文本/图片/文档多模态输入大模型动态切换通义千问、DeepSeek 等行业领域定制医疗、法律、金融SSE 流式输出使用 fetch-event-source今天我将逐行拆解核心代码带你复现这一高价值 AI 应用的前端架构。一、技术选型为什么是 fetch-event-source虽然浏览器原生支持 EventSource但在现代前端工程中我们更倾向使用 microsoft/fetch-event-source基于 fetch天然支持 AbortController可取消请求支持 自定义 headers用于鉴权更好的 TypeScript 类型支持与 Axios/Vite 生态无缝集成安装npm install microsoft/fetch-event-source二、OCR 图片识别前端只管“传”和“显”用户上传一张英文菜单截图系统自动提取文字并翻译。前端不碰 OCR 算法但要管理全流程状态。真实代码实现来自 文本_r9uv.txt1. 系统配置加载含文件大小限制// src/views/translation/index.vueconst loadSystemPublicConfig async () {try {const response await findSystemPublicConfig({ configKey: upload_file_max_limit_mb });const config response.data[0];systemConfig.value.uploadFileMaxLimit parseInt(config.value) || 10; // 默认10MB} catch (error) {ElMessage.error(加载系统配置失败);}};2. 文件选择与校验const handleFileSelect (event: Event) {const input event.target as HTMLInputElement;if (input.files input.files.length 0) {const file input.files[0];const maxSize systemConfig.value.uploadFileMaxLimit * 1024 * 1024;if (file.size maxSize) {message.error(文件不能超过 ${systemConfig.value.uploadFileMaxLimit} MB);return;}currentFile.value file;startParsing(file);}};3. 调用 OCR 接口Ant Design Vue 组件反馈const handleObtainFileContentText async () {if (!currentFile.value) return;const formData new FormData();formData.append(file, currentFile.value);try {const response await obtainFileContentText(formData);if (response.success) {sourceLanguageText.value response.data.contentText;currentUploadStatus.value success;} else {currentUploadStatus.value failed;message.error(response.message || 解析失败);}} catch (error) {currentUploadStatus.value failed;message.error(网络错误请重试);}};关键设计状态分为 idle | parsing | success | failed对应四个 UI 区块避免用户困惑。三、大模型动态集成让选择“无感而智能”AI 模型不是越多越好而是要默认最优 动态适配。真实代码实现1. 加载模型与语种字典// 加载大模型列表const loadLargeModelDictionary async () {const response await findLargeModelDictionary({ keyword: });largeModelList.value response.data;if (largeModelList.value.length 0) {const firstModel largeModelList.value[0];selectedModelId.value firstModel.id;await loadLargeModelLanguageSupport(firstModel.id); // 关键联动语种}};// 加载某模型支持的语种const loadLargeModelLanguageSupport async (modelId: string) {const response await findLargeModelLanguageSupport({ largeModelId: modelId });languageSupportList.value response.data;};2. 行业领域初始化const loadIndustrySectorDictionary async () {const response await findIndustrySectorDictionary({ keyword: });industrySectorList.value response.data;// 业务逻辑若默认为 general则跳过if (defaultIndustrySector.value general industrySectorList.value.length 1) {defaultIndustrySector.value industrySectorList.value[1].code;}};3. Ant Design Vue 下拉框绑定v-model:valueselectedModelIdstylewidth: 100%changehandleModelChangev-foritem in largeModelList:keyitem.id:valueitem.id{{ item.name }}四、SSE 流式翻译用 fetch-event-source 实现“打字机效果”这是体验升级的核心我们使用 microsoft/fetch-event-source 替代原生 EventSource。完整流式翻译实现结合你的项目结构import { fetchEventSource } from microsoft/fetch-event-source;// 存储控制器用于取消请求let abortController: AbortController | null null;const handleTranslate async () {if (isTranslating.value) return;// 重置结果translationResult.value ;isTranslating.value true;// 创建 AbortControllerabortController new AbortController();const params new URLSearchParams({largeModelId: selectedModelId.value,sourceLangCode: selectedSourceLangCode.value,targetLangCode: selectedTargetLangCode.value,industrySector: defaultIndustrySector.value,enableDeepThinking: String(enableDeepThinking.value),content: sourceLanguageText.value.trim(),});try {await fetchEventSource(/api/v1/large-model/translate/stream${params}, {method: GET,headers: {Authorization: Bearer ${getToken()}, // 从 Pinia 或 localStorage 获取},signal: abortAssistant.signal, // 支持取消onmessage(event) {if (event.data) {translationResult.value event.data;nextTick(() {scrollToBottom();});}},onclose() {isTranslating.value false;},onerror(err) {console.error(SSE Error:, err);message.error(翻译服务异常请稍后重试);isTranslating.value false;abortController.abort(); // 主动关闭},});} catch (error) {if (abortController.signal.aborted) {console.log(翻译已取消);} else {message.error(连接失败);}isTranslating.value false;}};// 取消翻译const cancelTranslation () {if (abortController) {abortController.abort();isTranslating.value false;message.info(翻译已取消);}};自动滚动到底部const scrollToBottom () {const container document.querySelector(.translation-result-content);if (container) {container.scrollTop container.scrollHeight;}};为什么不用 WebSocketSSE 是 单向流服务端 → 客户端完美匹配“AI 生成文本”场景fetch-event-source 提供 Promise 风格 API更符合现代前端习惯五、工程化亮点13年经验的沉淀1. 响应式布局PC/Mobile 适配const updateMainHeight () {const mainEl document.querySelector(.main) as HTMLElement;if (!mainEl) return;if (isMobile.value) {mainEl.style.minHeight 100vh;mainEl.style.height auto;} else {mainEl.style.height 100vh;mainEl.style.minHeight auto;}};2. 内存清理防止泄漏onUnmounted(() {if (abortController) {abortController.abort();}window.removeEventListener(resize, checkIsMobile);});3. 防重复提交 加载状态typeprimary:loadingisTranslatingclickhandleTranslate:disabled!sourceLanguageText.trim(){{ isTranslating 翻译中... : 开始翻译 }}结语前端的价值在“AI 与人之间”这个项目让我确信AI 时代前端工程师的不可替代性在于“体验设计”。我们用 fetch-event-source 把冰冷的 token 流变成温暖的“打字机”我们用 Ant Design Vue 让复杂配置变得简单我们用状态机管理 OCR 的每一步不让用户迷失。牙木顿衷

相关文章:

美团面试:为什么要用分布式缓存?本地缓存呢?多级缓存一致性如何保证?创

从 UI 工程师到 AI 应用架构者 13 年前,我的工作是让按钮在 IE6 上对齐; 13 年后,我用 fetch-event-source 订阅大模型的“思维流”,用 OCR 解锁图片中的文字——前端,正在成为 AI 产品的第一道体验防线。 最近&#x…...

Unity新手必看:如何用Input系统实现FPS游戏的键盘鼠标控制(附完整代码)

Unity FPS游戏开发实战:Input系统高级控制与优化技巧 第一次在Unity中尝试制作FPS游戏时,我花了两天时间才让角色不再像喝醉酒一样摇晃行走。键盘和鼠标输入的微妙配合、视角旋转的平滑处理、不同设备间的控制切换——这些看似基础的功能背后藏着许多新手…...

前端性能排查实战:Chrome Network面板里Timing那7个阶段到底怎么看?

Chrome Network面板Timing分析实战:从指标到性能优化 页面加载缓慢时,Chrome DevTools的Network面板中的Timing指标就像犯罪现场的指纹,每个数字背后都隐藏着性能问题的真相。但面对Queueing、Stalled、TTFB这些专业术语,很多开发…...

MySQL在事务中如何实现串行化_使用select lock in share mode查询

SELECT ... LOCK IN SHARE MODE 只阻塞其他事务的 SELECT ... FOR UPDATE 和 UPDATE/DELETE,不阻塞普通 SELECT 或其他共享锁;它允许多个事务同时读,但无法防止并发修改,需配合排他锁或原子更新使用。SELECT ... LOCK IN SHARE MO…...

COMSOL环偶极子增强磁光克尔效应

comsol环偶极子增强磁光克尔效应最近在玩COMSOL模拟磁光克尔效应的时候,发现环偶极子结构对增强效果特别有意思。这玩意儿就像给光波装了个磁力放大器,咱们今天直接上干货,看看怎么用COMSOL玩转这个现象。先搞明白环偶极子怎么在模型里构建。…...

SQL复杂数据聚合_嵌套子查询与GROUP BY配合

GROUP BY后不可直接选择未分组且未聚合的字段,MySQL 5.7和严格模式PostgreSQL会报错1055;正确做法是用子查询、窗口函数或ANY_VALUE()(需确认组内无差异),并注意NULL处理、索引优化与语义边界。GROUP BY 后不能直接选未…...

运算放大器电流流向的3个常见误区,硬件工程师必看避坑指南

运算放大器电流流向的3个常见误区,硬件工程师必看避坑指南 在硬件电路设计中,运算放大器(Op-Amp)作为模拟电路的核心器件,其电流流向的理解直接影响电路性能与稳定性。然而,即使是经验丰富的工程师&#xf…...

从聊天到办公全能:Kimi AI的隐藏功能大揭秘(含Prompt优化技巧)

从聊天到办公全能:Kimi AI的隐藏功能大揭秘(含Prompt优化技巧) 在AI工具井喷式发展的今天,Kimi AI凭借其独特的多场景适应能力,正在重新定义"智能助手"的边界。这款最初以聊天功能进入大众视野的工具&#x…...

**发散创新:基于Python的提示注入防御机制实战解析**在当前大模型广泛应用的时代,**提示注入(Promp

发散创新:基于Python的提示注入防御机制实战解析 在当前大模型广泛应用的时代,提示注入(Prompt Injection) 已成为不可忽视的安全风险。无论是API调用、Web应用集成还是本地部署的LLM服务,都可能因恶意构造输入而触发…...

**Bun运行时实战:用超快启动速度重构Node.js开发体验**在现代前端与后端协同开发中,*

Bun运行时实战:用超快启动速度重构Node.js开发体验 在现代前端与后端协同开发中,启动速度、开发效率和生态兼容性成为衡量一个运行时是否优秀的核心指标。近年来,Bun(https://bun.sh)作为一款新兴的JavaScript/TypeScr…...

西门子S7-200SMART与三菱变频器通讯程序:Modbus RTU协议下的高效控制解决方案

西门子S7-200SMART与三菱变频器通讯程序,实际效果如视频所示,认准店名未来电气,支持。 只是程序,不发快递物流,采用modbus rtu协议。 型号:plc西门子200smart,威纶通MT8071IE,变频器FR-E700(FR-…...

别再只用connectWifi了!微信小程序连接Wi-Fi的完整避坑指南(附getConnectedWifi实战代码)

微信小程序Wi-Fi连接全链路实战:从API陷阱到高可靠解决方案 每次看到connectWifi返回success却无法上网,或是onWifiConnected回调永远空数据时,作为开发者的你是否想砸键盘?微信小程序Wi-Fi模块的API设计就像个布满暗礁的航道——…...

从USB充电到HDMI传4K:聊聊PCB板上那些‘隐形’的100Ω和90Ω差分线

从USB充电到HDMI传4K:PCB板上那些‘隐形’的100Ω和90Ω差分线 当你用USB线给手机快速充电时,是否想过为什么有些充电线能稳定传输2.5A大电流?当你用HDMI线连接4K显示器时,是否疑惑过为什么画面从不闪烁?这些看似简单…...

宜搭高级认证考了3次才过?这份我踩过的坑和避坑指南请收好(含JS动作、集成自动化高频错题)

宜搭高级认证3次血泪史:JS动作与集成自动化高频错题深度拆解 第一次看到成绩单上"未通过"三个字时,我盯着屏幕发了十分钟呆——这已经是第二次失败了。作为有三年低代码开发经验的工程师,我原以为这种"拖拉拽"的认证考试…...

Ubuntu 20.04下VirtualBox USB设备识别全攻略:从增强包安装到用户组配置

Ubuntu 20.04与VirtualBox USB设备深度集成指南 在开发环境搭建过程中,我们经常需要在虚拟机中访问物理机的USB设备。Ubuntu 20.04 LTS作为长期支持版本,与VirtualBox的组合是许多开发者的首选方案。然而,当插入USB设备时,虚拟机却…...

别再为reg2icg的setup违例头疼了!手把手教你用ICC2/Innovus这3招搞定(附实战数据对比)

3大实战技巧彻底解决ICC2/Innovus中reg2icg的setup违例问题 在数字芯片后端设计中,时钟门控单元(ICG)与寄存器之间的时序路径(reg2icg)一直是工程师们最头疼的问题之一。特别是在先进工艺节点下,这类路径经常出现setup违例,直接影响芯片性能甚…...

新手避坑指南:用URDF给机械臂建模时,origin和inertial参数到底该怎么算?

机械臂URDF建模实战:origin与inertial参数计算完全指南 当你在Rviz中看到机械臂模型"飘在空中"或在Gazebo仿真时出现诡异抖动,八成是origin和inertial参数设置出了问题。这两个看似简单的参数,实则是URDF建模中最容易踩坑的"暗…...

保姆级教程:在vsomeip中为你的SOME/IP服务开启E2E保护(Profile 4配置详解)

深入实践:基于vsomeip的SOME/IP服务E2E保护配置全指南 在汽车电子系统开发中,功能安全始终是核心考量。当两个ECU通过SOME/IP协议通信时,如何确保消息在传输过程中不被篡改或丢失?这就是E2E(端到端)保护要解…...

机器学习40篇-开篇词-打通修炼机器学习的任督二脉

分享一个大牛的人工智能教程。零基础!通俗易懂!风趣幽默!希望你也加入到人工智能的队伍中来!请轻击人工智能教程​https://www.captainai.net/troubleshooter 在新进展层出不穷的今日,机器学习依然占据着人工智能的核心…...

[信号与系统]双线性变换在数字滤波器设计中的核心应用

1. 双线性变换:数字滤波器设计的桥梁 第一次接触数字滤波器设计时,我被一个核心问题困扰:如何把教科书上那些完美的模拟滤波器搬到计算机里运行?直到遇到双线性变换这个"魔法公式",才真正打通了模拟与数字世…...

PostgreSQL COPY命令实战:高效数据迁移与批量处理技巧

1. COPY命令基础:PostgreSQL的数据搬运工 第一次接触PostgreSQL的COPY命令时,我正面临着一个紧急的数据迁移任务。当时需要将百万级用户数据从旧系统迁移到新平台,试过各种方法后,COPY命令的导入速度让我震惊——比传统的INSERT语…...

第8篇 | Adaptive AUTOSAR的十字路口:高性能计算的标准化之路

当Classic Platform被形容为“精密的瑞士钟表”时,Adaptive Platform更像是“可扩展的云计算平台”。两者的哲学差异,决定了它们的应用边界。 Adaptive AUTOSAR核心模块 Adaptive平台引入的新模块: ara::com:服务发现与通信(SOME/IP、DDS可选)。 ara::exec:进程生命周期…...

别再只写Hello World了!用鸿蒙Next Beta2和DevEco Studio做个能存数据的通讯录

鸿蒙Next实战:从零构建具备数据持久化的智能通讯录应用 如果你已经厌倦了反复编写"Hello World"和各种静态界面演示,渴望在鸿蒙应用开发中实现真正的功能突破,那么构建一个完整的通讯录应用将是绝佳的进阶选择。本文将带你从零开始…...

【指数编制系列二】数据标准化方法实战:从理论到Python实现

1. 为什么需要数据标准化? 做过数据分析的朋友应该都遇到过这样的问题:当你试图把身高(厘米)和体重(千克)两个指标放在一起分析时,会发现身高的数值普遍比体重大几十倍。这时候如果直接计算两者…...

昇腾MindIE服务化推理实战:手把手教你用Qwen2-7B搭建高并发API服务(含代理避坑)

昇腾MindIE实战:Qwen2-7B模型高并发API服务部署全指南 当大语言模型从实验阶段走向生产环境,服务化部署成为技术落地的关键瓶颈。昇腾MindIE作为专为AI推理优化的服务化框架,通过continuous batching和PD分离架构等创新技术,显著提…...

零基础教程:用RetinaFace镜像快速搭建人脸检测与关键点绘制环境

零基础教程:用RetinaFace镜像快速搭建人脸检测与关键点绘制环境 1. 环境准备与快速部署 RetinaFace是目前最先进的人脸检测算法之一,能够同时实现高精度的人脸检测和关键点定位。本教程将带你从零开始,使用预置的RetinaFace镜像快速搭建完整…...

显示器色彩革命:novideo_srgb如何用NVIDIA显卡硬件校准解决广色域显示器过饱和问题

显示器色彩革命:novideo_srgb如何用NVIDIA显卡硬件校准解决广色域显示器过饱和问题 【免费下载链接】novideo_srgb Calibrate monitors to sRGB or other color spaces on NVIDIA GPUs, based on EDID data or ICC profiles 项目地址: https://gitcode.com/gh_mir…...

魔兽争霸3优化完全指南:让你的经典游戏在现代电脑上焕发新生

魔兽争霸3优化完全指南:让你的经典游戏在现代电脑上焕发新生 【免费下载链接】WarcraftHelper Warcraft III Helper , support 1.20e, 1.24e, 1.26a, 1.27a, 1.27b 项目地址: https://gitcode.com/gh_mirrors/wa/WarcraftHelper 还在为《魔兽争霸3》这款经典…...

保姆级教程:用PM2-Windows-Service将Node应用变成系统服务(含淘宝镜像加速)

保姆级教程:用PM2-Windows-Service将Node应用变成系统服务(含淘宝镜像加速) 在Windows服务器上部署Node.js应用时,最令人头疼的问题莫过于会话注销后应用进程自动终止。想象一下,你精心开发的在线商城后台服务&#x…...

FPGA新手必看:Xilinx GTX收发器VMGTAVCC供电设计避坑指南

Xilinx GTX收发器VMGTAVCC供电设计实战手册:从原理到避坑全解析 第一次接触Xilinx FPGA的GTX收发器设计时,我被VMGTAVCC这个看似普通的电源引脚折磨了整整两周。电路板上的眼图始终无法闭合,直到发现是去耦电容的布局犯了低级错误。这段经历让…...