阅读视图

发现新文章,点击刷新页面。

Tauri 踩坑 appLink 修改后闪退

如题,Grok 救我狗命,我得重新苹果 Grok 搜索能力的优势了。

**问题归根到底应该就是 Info.plist 没有在修改 appLink 之后自动更新,导致匹配不上。**解决方案是清理掉原来的自动生成内容,重新运行 ios init。估计不止这种问题,tauri + iOS 本来能正常跑,改了配置突然运行不了的情况,都可以尝试一下重新 init。这里还有点感悟,一旦出了舒适区,即使有 AI 也要花费不少时间解决这些问题,如果是熟练的 iOS 开发者应该一早就感应到这种问题了吧

1. “appLink”: false 的正确含义和用法

  • appLink: false(或省略) = 自定义 URI Scheme(如 myapp://xxx),不需要服务器、不需要 .well-known 文件。
  • 插件会在构建时自动往 Info.plist 里写入 CFBundleURLTypes
  • appLink: true(默认) = Universal Link / App Link(https + host),需要苹果验证文件。

正确配置(tauri.conf.json)

{
  "plugins": {
    "deep-link": {
      "mobile": [
        {
          "scheme": ["myapp"],        // ← 必须有这个,数组形式,方案名建议全小写
          "appLink": false            // 自定义 scheme 必须显式 false 或省略
        }
        // 如果你还想同时支持 Universal Link,可以再加一个对象
        // {
        //   "host": "yourdomain.com",
        //   "pathPrefix": ["/open"]
        //   // appLink 默认 true,可不写
        // }
      ]
    }
  }
}

常见写错导致闪退的几种情况

  • appLink: false 直接写在有 host 的对象里(混合使用)。
  • 只写 appLink: false 却没写 scheme(插件生成空/无效的 CFBundleURLTypes)。
  • 用错字段名:老版本可能是 "schemes",新版是 "scheme"(注意单数)。
  • appLink 写在 deep-link 顶层而不是 mobile 数组里的对象里。

2. 为什么会闪退?

  • Info.plist 生成出错(空 array、重复 key、XML 格式坏掉)。
  • iOS 系统在启动时解析 plist 失败,直接 kill 进程。
  • 之前用 Universal Link(appLink true)时 Xcode 已经加了 com.apple.developer.associated-domains entitlement,切换到 false 后 entitlement 和 provisioning profile 不匹配,也会导致启动失败。

3. 立即修复步骤(必做)

  1. 修正配置 为上面正确的格式(只保留 scheme + appLink: false)。
  2. 彻底清理构建缓存
    tauri ios clean          # 如果有这个命令
    # 或者手动
    rm -rf target/ src-tauri/target src-tauri/gen/apple
    
  3. 删除 Xcode DerivedData(最重要):
    rm -rf ~/Library/Developer/Xcode/DerivedData
    
  4. 重新生成/运行:
    tauri ios dev            # 或 tauri ios build
    
  5. 如果还是闪退,打开 Xcode 项目(target/ios 里生成的 .xcodeproj),找到 Info.plist,搜索 CFBundleURLTypes,确认里面有你的 scheme,且没有空数组。

4. 验证是否成功

  • 构建成功后,用模拟器测试:
    xcrun simctl openurl booted myapp://test
    
  • App 应该正常启动并收到 deep-link 事件。

总结:99% 是配置写错了(少了 scheme 或混用了 host),导致 Info.plist 坏掉。改对配置 + 清理缓存基本就能解决。如果清理后还闪退,把你的 plugins.deep-link 那段完整配置贴出来,我再帮你看。

又被 Safari 差异坑了:textContent 拿到的值居然没换行?

作为前端开发,我们习惯了用 textContent 来获取纯文本。毕竟它性能好、不触发重排(Reflow),是 W3C 标准的亲儿子。但今天,我在处理一个自动换行的多行文本时,被 Safari 给上了一课。

起因:消失的换行符

需求很简单:用户在一个高度自适应的 <div> 中输入或展示一段文字,我需要抓取这段文字存入数据库。

为了追求性能,我习惯性地写下了:

const content = document.querySelector('.text-box').textContent;

ChromeFirefox 下,一切完美。由于 CSS 设置了 white-space: pre-wrap;,我拿到的字符串里自带优雅的 \n

然而,测试同学拿着 Safari 跑过来:“为什么存进去的数据全挤成一团了?”


破案:textContent 并不“所见即所得”

经过一番排查,我发现了这个隐藏极深的坑点:

1. textContent 的“冷酷”

textContent 获取的是 DOM 树中所有文本节点的原始数据。它根本不在乎你的 CSS 长什么样。

  • 如果你的 HTML 源码里是一行,即便 CSS 用 word-breakwhite-space 让它在视觉上换了行,textContent 拿到的依然是硬邦邦的一行

2. Safari 的“严格”

在某些特定版本的 Safari 渲染引擎中,它对 textContent 的实现非常遵循“源码原始性”。如果文本是因为容器宽度挤压产生的软换行(Soft Wrap),Safari 的 textContent 绝对不会帮你补上那个 \n


解决方案:innerText 才是救星

这时候,那个曾经被嫌弃“性能略差”的 innerText 站了出来。

为什么这次要用 innerText?

  • 感知 CSS 样式innerText 是受 CSS 影响的。它会触发一次布局计算,获取用户肉眼看到的文本形态。
  • 自动转换换行:如果你的元素里有 <br>,或者因为 white-space: pre-wrap 产生了视觉换行,innerText 会非常贴心地在对应位置插入 \n

代码修正:

// ❌ 坑点代码(Safari 下可能丢失换行)
// const text = el.textContent;

// ✅ 避坑代码(所见即所得,保留视觉换行)
const text = el.innerText;


总结:避坑指南

这次折腾让我记住了两点:

  1. 如果你要的是“源码里的字”:选 textContent,它快且稳,不理会 CSS。
  2. 如果你要的是“屏幕上的字”:特别是涉及**换行、空格、大小写转换(text-transform)**时,请务必果断使用 innerText

前端无小事,永远不要低估 Safari 的“独特性”。 以后看到文本抓取需求,还是先老老实实测一下兼容性吧!


📝 避坑速查表

场景 推荐属性 原因
高性能纯文本提取 textContent 不触发重排
保留 <br> 换行 innerText 会将标签转为 \n
处理隐藏元素 textContent innerText 拿不到隐藏文本
获取视觉换行文本 innerText 解决 Safari 差异的关键

看完就懂 useSyncExternalStore

功能

React 引入 useSyncExternalStore 也很长一段时间了,但是存在感还不太强。简而言之,它专门用来搞定那些不受 React 内部生命周期控制的外部数据源

过去最大的问题其实是 React 渲染时的 「撕裂」,这是 React 为了优化页面响应速度引入的并发渲染机制带来的副作用。

简单来说就是 React 为了防止在渲染时长时间无法响应用户输入,把渲染过程拆分成多个可中断的小任务,这就能小任务的间隙中插入用户响应,从而模拟出「并发」的感觉。更完整的前因后果可以参考《React 的设计哲学》

在 React 并发渲染机制下,如果用普通的 useEffect 去同步外部数据,可能会出现渲染进行到一半时数据突然发生变化,导致同一份页面中,一半的组件拿着老数据,另一半拿着新数据的灵异现象(但是实际上出现这个问题的几率其实非常小,大家都忽略了,这就导致了 useSyncExternalStore 的存在感很低)。使用 useSyncExternalStore 后,如果在渲染过程中快照发生变化,React 会丢弃当前渲染并重新开始,从而保证同一次提交中的所有组件看到的是同一个版本的数据。

使用场景

订阅浏览器 API

拿监听网络状态来说。不使用这个 Hook 之前,我们通常得在组件里写个包含完整挂载和清理逻辑的 useEffect 去监听 onlineoffline 事件。

function subscribe(callback) {
  window.addEventListener("online", callback);
  window.addEventListener("offline", callback);
  return () => {
    window.removeEventListener("online", callback);
    window.removeEventListener("offline", callback);
  };
}

function getSnapshot() {
  return navigator.onLine;
}

// 组件里直接这么用
const isOnline = useSyncExternalStore(subscribe, getSnapshot);

监听媒体查询(Media Queries)响应式布局也是同样的套路:

const query = window.matchMedia("(max-width: 600px)");

function subscribe(callback) {
  query.addEventListener("change", callback);
  return () => query.removeEventListener("change", callback);
}

const isMobile = useSyncExternalStore(subscribe, () => query.matches);

轻量级全局状态

如果你接手了一个极小的项目,不想引入 Redux 或 Zustand 这样繁琐的包,但又迫切需要在几个跨层级的组件间共享某部分状态。这时候你可以直接手搓一个简易的 Store:

// 丢在 React 外面的状态中心
let internalState = { count: 0 };
const listeners = new Set();

const store = {
  increment() {
    internalState = { count: internalState.count + 1 };
    listeners.forEach((l) => l());
  },
  subscribe(callback) {
    listeners.add(callback);
    return () => listeners.delete(callback);
  },
  getSnapshot() {
    return internalState;
  },
};

// 任何组件里都可以直接同步获取状态
const state = useSyncExternalStore(store.subscribe, store.getSnapshot);

注意:useSyncExternalStore 内部用 Object.is 比较前后快照,如果 getSnapshot 在数据未变的情况下每次都返回新对象,会导致无限循环重渲染。

只要把这段代码看懂,你就掌握了 Zustand 这种现代状态管理库的核心原理

竞品 API

useEffect + setState

曾经大家都习惯在 useEffect 里监听外部变化,如果变了,再跑一下 setState 触发更新。

这就又到了日常批判 useEffect 的时候了。

useEffect 带来重复渲染和闪烁问题。如果你的外部状态和页面初始计算的状态不对齐,页面渲染就会经历「旧值 -> 闪烁 -> 新值」这三步。而 useSyncExternalStore 在渲染中途就能直接取走最新的正确值。

另外,在处理服务端渲染时,用副作用很容易抛出水合(Hydration)错误,因为服务端和客户端首次生成的 HTML 大概率因为外部数据对不上。useSyncExternalStore 为此专门开了一个叫 getServerSnapshot 的参数,让你传能兜底服务端的静态快照。

Context

很多人滥用 Context 做全局状态,但如果是频繁变动的数据,Context 的广播机制简直是一场灾难。只要 Provider 提供的值发生了变动,它底下所有的子组件也会跟着无脑重跑 Render,除非你给每个组件层级套一层 React.memo(当然现在有 compiler,但也不是毫无代价)。

相比之下,useSyncExternalStore 实现了高精度的按需订阅——只有从 Store 取出的快照真的有了变化,关联的组件才会再次渲染。在这里还是顺便强调一下,没事别用 Context。

总结

要判断何时使用 useSyncExternalStore 其实很简单,只要你的数据依然在 React 的生命周期里流转(例如表单实时输入、控制弹窗开闭的布尔值),那就老老实实用回你的 useStateuseReducer

一旦数据满足游离于 React 管理之外、会随时间变化、且你要让 UI 能自动响应这种变化这三个条件,就毫不犹豫上 useSyncExternalStore。日常写前端页面也许碰不到几次,但之后你要是去造底层 Hook 库,或者需要硬啃第三方库内部暴露出的状态时,useSyncExternalStore 绝对好使~

相关链接

❌