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

山东大学软件学院项目实训-创新实训-大数据租房推荐智能体-前端部分(3)

虽然上一阶段搞定了“打字机”效果让 AI 看起来反应很快但我发现了一个新问题光有文字看房体验还是很累。所以这一阶段的目标很明确正如上一篇博客提到的下一阶段目标我要把 AI 的回复从“纯文本”升级成“富媒体卡片”。1. 遇到的难题流式传输 vs 结构化数据这其实是本项目最难的一个技术点。之前我们用st.write_stream来显示打字机效果但它有个限制它只能接收字符串。而房源卡片需要的是结构化的数据比如{title: 阳光小区, price: 3500, img: ...}。如果直接把 JSON 数据混在文字流里吐出来页面上就会显示一堆乱码代码用户体验极差。2. 核心解法神奇的“旁路传输”经过一番调研我设计了一个“旁路传输”的方案。既然yield只能传文字给前端我于是在 Python 后端搞一个“容器”List。主路文字AI 继续yield文本保证页面上的字是一个个蹦出来的。旁路数据AI 在后台解析数据时如果发现有房源信息就把它append进那个“容器”里。这样等文字流结束了我的“共享容器”里也装满了数据直接拿去渲染卡片就行3.网络层的封装为了配合这种机制我优化了app_lib/agent_client.py。这里的核心是post_sse_line_iter函数。它不再直接返回解析好的文本而是返回原始的 SSE 行数据。这样做的好处是解耦——具体的解析逻辑判断哪一行是文本哪一行是房源 JSON被下放到了具体的业务逻辑中而不是写死在网络请求库里。def post_sse_line_iter( url: str, json_body: Mapping[str, Any] | None None, *, headers: Mapping[str, str] | None None, connect_timeout: float DEFAULT_CONNECT_TIMEOUT, read_timeout: float DEFAULT_READ_TIMEOUT, ) - Iterator[str]: timeout httpx.Timeout( connectconnect_timeout, readread_timeout, write30.0, pool30.0, ) client httpx.Client(timeouttimeout) try: with client.stream( POST, url, jsondict(json_body) if json_body is not None else {}, headers{ Accept: text/event-stream, **(dict(headers) if headers else {}), }, ) as response: response.raise_for_status() for line in response.iter_lines(): if line is not None and line ! : yield line finally: client.close()同时build_chat_payload负责把前端的对话历史包装成后端需要的格式。这里我处理了history只保留user和assistant的角色防止系统指令污染上下文4. UI 组件化打造房源栅格系统有了数据接下来就是怎么展示。我把所有与 UI 相关的逻辑都抽离了app_lib/listing_cards.py在_render_one_card中我处理了大量的边界情况。比如缺失数据处理如果房源没有价格或面积不能直接报错而是显示“—数据待补充”。链接去重_listing_links函数会检查多个链接链家、贝壳、外链自动去重并动态生成st.link_button。图片懒加载虽然Streamlit的st.image很简单但我还是加了判断如果没有thumb_url则显示占位提示。def _render_one_card(li: Listing, *, key_prefix: str) - None: title (li.get(title) or ).strip() or 标题待补充 st.markdown(f**{title}**) price li.get(price) area li.get(area_sqm) col_price, col_area st.columns(2) with col_price: if isinstance(price, (int, float)): st.caption(_fmt_line(总价, f{price:.0f} 万)) else: st.caption(_fmt_line(总价, None)) with col_area: if isinstance(area, (int, float)): st.caption(_fmt_line(面积, f{area:.1f} ㎡)) else: st.caption(_fmt_line(面积, None)) st.caption(_fmt_line(户型, li.get(rooms))) st.caption(_fmt_line(楼层, li.get(floor))) st.caption(_fmt_line(朝向, li.get(orientation))) bits [] if comm : li.get(community): bits.append(_fmt_line(小区, comm).replace(小区, ).strip()) if dist : li.get(district): bits.append(_fmt_line(区域, dist).replace(区域, ).strip()) if bits: st.caption( · .join(bits)) else: st.caption(小区 / 区域—数据待补充) score li.get(score) if isinstance(score, (int, float)): sf float(score) st.progress(min(1.0, max(0.0, sf / 10.0))) st.caption(f参考分 {sf:.1f}/10条形占位详细图表见第 6 段) else: st.caption(评分—待第 6 段图表接入) thumb li.get(thumb_url) if thumb: st.image(str(thumb), use_container_widthTrue) else: st.caption(缩略图—大图懒加载占位) with st.expander(详情与外链, expandedFalse): dec li.get(decoration) st.write(_fmt_line(装修, dec if isinstance(dec, str) else None)) _listing_links(li) fid str(li.get(listing_id, )) bk _safe_key(key_prefix, fav, fid) ck _safe_key(key_prefix, cmp, fid) if st.button(收藏, keybk, use_container_widthFalse): fav st.session_state.setdefault(favorite_listing_ids, set()) if fid in fav: fav.discard(fid) st.toast(已取消收藏) else: fav.add(fid) st.toast(已加入收藏) if st.button(加入对比, keyck, use_container_widthFalse): cmp_ids st.session_state.setdefault(compare_listing_ids, []) if fid and fid not in cmp_ids: cmp_ids.append(fid) st.toast(已加入对比入口预留)栅格布局与滚动优化为了模拟真实App的列表感我使用了st.columns(3)来实现三列栅格。在render_listing_grid中实现了滚动区。为了避免页面无限拉长我通过st.markdown注入了一段HTML/CSS限制了房源列表的最大高度max-height: 720px并开启overflow-y: auto。这样当推荐房源过多时卡片区域会出现独立的滚动条而不会影响整个聊天界面的布局。ncols 3 st.markdown( fdiv stylemax-height:{scroll_max_height_px}px;overflow-y:auto;padding-right:6px;, unsafe_allow_htmlTrue, ) idx 0 while idx len(visible): cols st.columns(ncols) for c in range(ncols): if idx len(visible): break with cols[c]: with st.container(borderTrue): _render_one_card(visible[idx], key_prefix_safe_key(grid_key, str(idx))) idx 1 st.markdown(/div, unsafe_allow_htmlTrue)通过这种方式我实现了“字是一个个打出来的卡片是随后整齐排列的”这一交互。5. 总结与展望现在我们的智能体不仅能“陪聊”还能“办事”了虽然都是前端的模拟流式输出。代码结构上agent_client负责路listing_cards负责车app.py负责调度分层非常清晰。随着房源越看越多我意识到现在的对话历史存在一个致命弱点刷新即焚。一旦用户不小心刷新了浏览器之前辛苦筛选的房源和聊过的需求就会全部丢失所以在下一阶段我准备引入会话历史持久化与「50 条」策略深化这一阶段内容。

相关文章:

山东大学软件学院项目实训-创新实训-大数据租房推荐智能体-前端部分(3)

虽然上一阶段搞定了“打字机”效果,让 AI 看起来反应很快,但我发现了一个新问题:光有文字,看房体验还是很累。所以,这一阶段的目标很明确,正如上一篇博客提到的下一阶段目标:我要把 AI 的回复从…...

从标准库到HAL库:手把手移植STM32 Modbus-RTU代码的避坑指南

从标准库到HAL库:STM32 Modbus-RTU移植的深度实践 当我们需要将现有的STM32标准库Modbus-RTU项目迁移到HAL库时,这个过程远比简单的函数替换复杂得多。本文将深入探讨移植过程中的关键差异点、常见陷阱以及解决方案,帮助开发者顺利完成这一技…...

FPGA开发者必看:手把手教你用Verilog实现HDMI 1.4视频输出(基于Zynq 7020)

FPGA实战:基于Zynq 7020的HDMI 1.4发送器全流程开发指南 当我们需要在Zynq 7020的PL端实现HDMI输出时,面临的第一个挑战是如何将协议文档中的理论转化为可综合的RTL代码。本文将带你从TMDS编码器设计开始,逐步构建完整的HDMI发送系统&#xf…...

漫画翻译革命:如何用BallonsTranslator让外文漫画阅读零门槛?

漫画翻译革命:如何用BallonsTranslator让外文漫画阅读零门槛? 【免费下载链接】BallonsTranslator 深度学习辅助漫画翻译工具, 支持一键机翻和简单的图像/文本编辑 | Yet another computer-aided comic/manga translation tool powered by deeplearning …...

从“按钮变色”到“文本互动”:用Tkinter StringVar改造你的第一个GUI小游戏

从“按钮变色”到“文本互动”:用Tkinter StringVar改造你的第一个GUI小游戏 当你第一次用Tkinter做出那个点击按钮会变色的程序时,那种成就感可能还记忆犹新。但很快你会发现,真正的GUI应用远不止于此——用户输入、动态反馈、状态更新才是交…...

Mapbox踩坑实录:图层叠加、图片更新、弹窗样式,这些坑我帮你填平了

Mapbox实战避坑指南:图层管理、动态图片与弹窗优化 第一次在项目中集成Mapbox时,那种兴奋感很快被各种意想不到的报错消磨殆尽。记得凌晨三点调试updateImage方法时,控制台不断抛出"Image dimensions must match"的错误——原来只是…...

Flux Sea Studio 跨平台渲染方案:云端生成与本地预览的协同

Flux Sea Studio 跨平台渲染方案:云端生成与本地预览的协同 最近在折腾一些创意项目时,我遇到了一个挺普遍的问题:手头的设计工具,要么功能强大但只能在特定设备上跑,对硬件要求高得吓人;要么就是能跨平台…...

别再傻等GitHub了!用Gitee镜像5分钟搞定Nacos 1.4.0源码编译与启动

国内开发者福音:5分钟极速搭建Nacos 1.4.0开发环境实战指南 每次打开GitHub准备下载Nacos源码时,那个缓慢的进度条是否让你抓狂?特别是在紧急修复线上问题或赶项目进度时,等待源码下载的时间简直让人崩溃。作为国内开发者&#x…...

用舞蹈链(DLX)算法搞定数独和八皇后:从理论到C++实战避坑

舞蹈链算法实战:用DLX高效解决数独与八皇后问题 第一次接触精确覆盖问题时,我正被一道"魔鬼级"数独题折磨得焦头烂额。传统回溯算法在9x9的网格中显得力不从心,直到发现了Donald Knuth提出的舞蹈链(Dancing Links&#…...

从M3U8密钥到DRM:实战解析主流流媒体视频加密方案

1. 从M3U8文件看流媒体加密基础 第一次接触M3U8文件时,我盯着那些以#EXT开头的标签看了半天,感觉就像在破解某种神秘代码。后来才发现,这其实是HLS(HTTP Live Streaming)协议的核心部分。简单来说,M3U8就是…...

游戏开发新思路:用SDF实现超低开销的软阴影与AO(以Bunny模型为例)

游戏开发新思路:用SDF实现超低开销的软阴影与AO(以Bunny模型为例) 在独立游戏开发中,画面表现与性能开销往往难以兼得。传统阴影和环境光遮蔽(AO)方案如Shadow Map和SSAO虽然效果尚可,但对硬件资…...

突破传统限制:ESP-SR离线语音识别框架的实战创新指南

突破传统限制:ESP-SR离线语音识别框架的实战创新指南 【免费下载链接】esp-sr Speech recognition 项目地址: https://gitcode.com/gh_mirrors/es/esp-sr ESP-SR是乐鑫科技专为ESP32系列芯片优化的嵌入式智能语音识别框架,提供完全离线的语音识别…...

Display Driver Uninstaller:3层深度清理技术解析与显卡驱动冲突解决方案

Display Driver Uninstaller:3层深度清理技术解析与显卡驱动冲突解决方案 【免费下载链接】display-drivers-uninstaller Display Driver Uninstaller (DDU) a driver removal utility / cleaner utility 项目地址: https://gitcode.com/gh_mirrors/di/display-dr…...

哔哩下载姬终极指南:5分钟快速掌握B站视频高效下载技巧

哔哩下载姬终极指南:5分钟快速掌握B站视频高效下载技巧 【免费下载链接】downkyi 哔哩下载姬downkyi,哔哩哔哩网站视频下载工具,支持批量下载,支持8K、HDR、杜比视界,提供工具箱(音视频提取、去水印等&…...

从零理解软件无线电:用GNU Radio仿真带你搞懂AM调制与解调全过程

从零理解软件无线电:用GNU Radio仿真带你搞懂AM调制与解调全过程 在通信工程领域,软件无线电(SDR)技术正以前所未有的方式重塑着信号处理的边界。不同于传统硬件无线电设备需要专用电路实现每个功能模块,SDR将大部分处…...

别再source错了!ROS2工作空间环境变量配置保姆级避坑指南(含ROS1/ROS2共存场景)

ROS2工作空间环境变量配置全攻略:从基础到多版本共存实战 每次打开终端都要source环境变量?ROS1和ROS2的命令总是冲突?工作空间里的包莫名其妙被覆盖?如果你正在经历这些困扰,这篇文章将彻底解决你的痛点。作为机器人…...

别再死磕PID了!用Python+scikit-fuzzy手把手教你实现一个智能水箱水位模糊控制器

用Pythonscikit-fuzzy实现智能水箱水位模糊控制器:超越PID的实践指南 水位控制是工业和生活场景中的常见需求,从家庭热水器到大型水处理厂都离不开这一基础控制环节。传统PID控制器虽然简单可靠,但在面对非线性、时变或存在不确定性的系统时&…...

2026届学术党必备的AI学术方案推荐

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 当下市场里主流的AI论文写作辅助工具无不各有侧重,在文献检索跟总结方面&#xf…...

从零到精通:AI大模型的全方位学习路径解析

本文深入解析了人工智能领域的大型预训练模型(大模型),将其比作“超级大脑”,通过海量信息学习世界知识,并详细阐述了学习大模型的重要性和广泛应用场景,如自然语言处理、内容推荐、教育、医疗、商业分析等…...

从零到一:在IDEA中高效配置Lua开发环境(解释器+插件实战)

1. 为什么选择IDEA开发Lua? 很多刚接触Lua的开发者会纠结该用什么开发工具。记事本太原始,专用Lua IDE又太重,而IDEA恰好是个折中的完美选择。我最初用Sublime Text写Lua,后来切换到IDEA,最大的感受就是代码提示和调试…...

本地LLM部署:硬件配置指南

文章主要探讨了自托管 AI 的优势及必要性,详细分析了与 AI 相关的关键硬件组件,包括 GPU、RAM、CPU 和 SSD,并强调了显存(VRAM)在 LLM 推理中的核心作用。文章还提供了从入门到发烧的硬件配置建议,如 Ollam…...

UML和面向对象

UML(统一建模语言,Unified Modeling Language)和面向对象(Object-Orientation)是软件工程中紧密相连的两个概念。面向对象是一种程序设计思想,而 UML 是一种可视化建模语言,用于表达面向对象分析(OOA)与设计(OOD)的成果。两者结合,使复杂系统的分析、设计、沟通和文…...

3个实战技巧让你高效掌握Chrome二维码插件的必备功能

3个实战技巧让你高效掌握Chrome二维码插件的必备功能 【免费下载链接】chrome-qrcode chrome-qrcode - 一个 Chrome 浏览器插件,可以生成当前 URL 或选中文本的二维码,或解码网页上的二维码。 项目地址: https://gitcode.com/gh_mirrors/ch/chrome-qrc…...

告别模拟器:用Termux+Ubuntu+JDK在安卓手机上搭建轻量Java开发环境

安卓手机变身Java开发机:TermuxUbuntuJDK全栈解决方案 在咖啡馆等朋友时突然需要调试一段业务逻辑代码,出差途中发现线上服务报错需要紧急修复,通勤路上想继续昨晚未完成的算法练习——这些场景下,我们往往懊恼没带笔记本电脑。其…...

G-Helper:重新定义华硕笔记本性能控制的轻量级革命

G-Helper:重新定义华硕笔记本性能控制的轻量级革命 【免费下载链接】g-helper Lightweight, open-source control tool for ASUS laptops and ROG Ally. Manage performance modes, fans, GPU, battery, and RGB lighting across Zephyrus, Flow, TUF, Strix, Scar,…...

2026年安卓反调试安全加固公司怎么选?从防Frida到上架审核全维度对比

当你的安卓应用核心算法、支付协议或通信密钥面临被逆向破解的风险时,找到一家真正靠得住的反调试加固公司就成了决定产品生死的关键选择题。这不是简单的采购,而是一次高风险的技术选型。市面上打着“安全加固”旗号的服务商不少,但真正能防…...

如何高效使用Markdown Viewer浏览器插件:掌握专业文档预览的5个核心技巧

如何高效使用Markdown Viewer浏览器插件:掌握专业文档预览的5个核心技巧 【免费下载链接】markdown-viewer Markdown Viewer / Browser Extension 项目地址: https://gitcode.com/gh_mirrors/ma/markdown-viewer 还在为浏览器中无法优雅预览Markdown文档而烦…...

从CI/CD流水线故障排查说起:当git pull显示已更新,但服务器文件纹丝不动时怎么办?

从CI/CD流水线故障排查说起:当git pull显示已更新,但服务器文件纹丝不动时怎么办? 在自动化部署的世界里,最令人抓狂的莫过于明明看到git pull输出"Already up-to-date",却发现服务器上的代码纹丝未动。这种…...

用Verilog和有限状态机(FSM)设计一个浪漫的8路流水灯(附完整代码与Quartus II仿真)

用Verilog和有限状态机打造浪漫的8路流水灯:从技术到情感的电子情书 当冰冷的电路遇上温暖的情感,技术便有了灵魂。想象这样一个场景:在特殊的日子里,你亲手设计的LED灯带缓缓亮起,从两端向中心汇聚的光芒如同两颗逐渐…...

Degrees of Lewdity汉化版完整指南:5分钟完成中文游戏配置

Degrees of Lewdity汉化版完整指南:5分钟完成中文游戏配置 【免费下载链接】Degrees-of-Lewdity-Chinese-Localization Degrees of Lewdity 游戏的授权中文社区本地化版本 项目地址: https://gitcode.com/gh_mirrors/de/Degrees-of-Lewdity-Chinese-Localization …...