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

张开发
2026/4/20 14:02:42 15 分钟阅读

分享文章

山东大学软件学院项目实训-创新实训-大数据租房推荐智能体-前端部分(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 条」策略深化这一阶段内容。

更多文章