96SEO 2026-06-21 11:36 1
一条消息在任意时刻只会处于一个状态 —— 那就用一个字段、一个枚举来表达它。这在软件设计里有句口号叫 "让非法状态无法被表示"
AI 对话比微信还多两步 —— AI 要"思考"、要"一个字一个字往外蹦"、你还Neng中途喊停。状态geng多,记录方式就geng容易写乱。这篇就从我 demo 里一段"kan着Neng跑、其实埋雷"的旧代码讲起。

@Trace isLoading: boolean = false // 整个会话共用一个
然后所有跟状态有关的事,dou用hen"土"的方式硬塞:
@ObservedV2export class ChatMessage { @Trace content: string = '' @Trace isLoading: boolean = false // 思考中 @Trace isStreaming: boolean = false // 流式输出中 @Trace isFailed: boolean = false // 失败了 @Trace stoppedMessage: string = '' // 停止提示}
比第一版强多了 —— 至少状态从正文里分出来了。但它有个隐患,有个专门的名字叫 布尔陷阱 / 布尔汤
msg.isLoading = truemsg.isFailed = true // 又在思考、又失败了?这是什么状态?🤔
msg.isStreaming = truemsg.isLoading = true // 既在思考又在流式?自相矛盾
这些组合在类型上完全合法,Neng编译、Neng赋值,但语义上是坏数据。一旦哪段逻辑漏改一个开关,UI 就会进入一个"谁dou没设计过"的中间态 —— 这类 bug Zui难查,因为它本不该存在。
为什么不用多个布尔值表示状态?说实话,多个布尔值kan起来挺直观,但维护起来真心头疼 —— 你得时刻记着哪些开关Neng同时开,哪些互相排斥。
@ObservedV2export class ChatMessage { @Trace content: string = '' // 只装正文 role: string = '' @Trace status: MessageStatus = MessageStatus.DONE // 状态独立成字段}
对比一下三版的差距:
hen自然的下一步,是给消息加一排开关,生产里不少代码也是这么写的:
Zui早我的消息模型长这样,只有内容,没有"状态"这个概念:
优化消息状态管理"正在生成"靠 ViewModel 上一个全局布尔:这是这次Zui想纠正的一个坏习惯。回kan旧代码:顺手把所有界面文案收口到一个常量类,告别魔法字符串散落: 害,为啥非得纠正呢?你猜。
"让非法状态无法被表示"// chat/src/main/ets/constants/ChatConstants.etsexport class ChatText { static readonly THINKING: string = '思考中' static readonly STOPPED: string = 'Yi停止生成' static readonly RESEND: string = '重新发送' static readonly REGENERATE: string = '重新生成' // ...}
UI 按状态渲染:ArkUI V2 有个坑 —— "响应式字段必须在 build 里读"
build { Column { // ✅ 直接在 build 里读 this.msg.status,依赖被追踪,状态一变就重渲染 if { Row { LoadingProgress.width.height.color Text.fontColor } } else if { // 流式光标:用一个 Span 拼,仍然不进 content Text { Span Span .fontColor } } }}
"顶层读字段",而不是传参给函数 —— 这是 V2 的限制,别绕弯子!"为什么百度不收录我的文章?" —— 有人问。 咱就是说这事儿跟你文章质量、关键词布局dou有关系。 你得确保内容足够有价值,还要优化好 meta 标签,让搜索引擎爱上你 🤣 。 不是有句话叫 "Content is King" 么?内容为王嘛! 对了还有外链啥的,也不Neng忽视,你懂的。
"停止生成""失败重试"的 UI 怎么画?—— kankan新代码怎么解耦!⚠️ 反例:
MyBuilder把响应式字段当@Builder入参传进去 —— V2 收不到依赖,status变了这块 UI 纹丝不动。记住:结构性的if/.visibility/表达式,直接读字段,别绕一层函数参数。
// ❌ 旧:状态拼进正文
this.activeAiMessage.content = old + '
'
// ✅ 新:状态归状态,正文归正文this.activeAiMessage.status = MessageStatus.STOPPED// content 不动,保持用户停止前Yi收到的部分,一个字不改
"一条消息的状态变迁图,其实就是需求的另一种画法。"
用户消息:
SENDING ──送达──► DONE
└──发不出去 / 服务端报错──► FAILED
AI 消息:
THINKING ──首个 chunk──► STREAMING ──流结束──► DONE
│ ├─用户点停止─► STOPPED
│ └─中途断网───► FAILED
└── 一个字没来 / 直接报错 ─────────────────► FAILED
举个例子,这张图列出了所有可Neng的状态和迁移路径。
对照需求,每个状态dou有了明确归宿:
// chat/src/main/ets/models/MessageStatus.etsexport enum MessageStatus {
SENDING = 'sending', // 用户消息发出去了但还没收到服务端确认
THINKING = 'thinking', // AI 开始处理用户请求,等首个字
STREAMING = 'streaming', // AI 返回结果逐字输出中
DONE = 'done', // 消息正常完成
STOPPED = 'stopped', // 用户主动停止 AI 输出
FAILED = 'failed', // 消息失败
}
项目改造目标:
- 文件1:models/MessageStatus.ets
- 文件2:models/chatModel.ets
- 文件3:components/ChatListComp.ets
一句话:
把「这条消息现在是什么状态」从散落的几个布尔 + 字符串拼正文,升级成一个枚举变量。
这是本系列“对话里的那些状态”的第一篇,主要讲数据建模,后面会聊编排和交互逻辑 👉 状态机驱动 UI,多么优雅 😎 。
作为专业的SEO优化服务提供商,我们致力于通过科学、系统的搜索引擎优化策略,帮助企业在百度、Google等搜索引擎中获得更高的排名和流量。我们的服务涵盖网站结构优化、内容优化、技术SEO和链接建设等多个维度。
| 服务项目 | 基础套餐 | 标准套餐 | 高级定制 |
|---|---|---|---|
| 关键词优化数量 | 10-20个核心词 | 30-50个核心词+长尾词 | 80-150个全方位覆盖 |
| 内容优化 | 基础页面优化 | 全站内容优化+每月5篇原创 | 个性化内容策略+每月15篇原创 |
| 技术SEO | 基本技术检查 | 全面技术优化+移动适配 | 深度技术重构+性能优化 |
| 外链建设 | 每月5-10条 | 每月20-30条高质量外链 | 每月50+条多渠道外链 |
| 数据报告 | 月度基础报告 | 双周详细报告+分析 | 每周深度报告+策略调整 |
| 效果保障 | 3-6个月见效 | 2-4个月见效 | 1-3个月快速见效 |
我们的SEO优化服务遵循科学严谨的流程,确保每一步都基于数据分析和行业最佳实践:
全面检测网站技术问题、内容质量、竞争对手情况,制定个性化优化方案。
基于用户搜索意图和商业目标,制定全面的关键词矩阵和布局策略。
解决网站技术问题,优化网站结构,提升页面速度和移动端体验。
创作高质量原创内容,优化现有页面,建立内容更新机制。
获取高质量外部链接,建立品牌在线影响力,提升网站权威度。
持续监控排名、流量和转化数据,根据效果调整优化策略。
基于我们服务的客户数据统计,平均优化效果如下:
我们坚信,真正的SEO优化不仅仅是追求排名,而是通过提供优质内容、优化用户体验、建立网站权威,最终实现可持续的业务增长。我们的目标是与客户建立长期合作关系,共同成长。
Demand feedback