暗无天日

=============>DarkSun的个人博客

MobileOrg Android:从 API 17 迁移到 API 34 的实战记录

本文迁移全程借助 Claude Code 完成,我本人没有手动编写任何代码。整个过程是:真机测试发现问题 → 将 logcat 或现象描述给 Claude Code → Claude Code 定位原因并生成修复代码 → 推送 GitHub Actions 构建 → 再测试验证。包括本文本身,也是借助 Claude Code 生成的。

由于我不是 Android 开发者,对文中涉及的技术细节无法保证完全准确,请读者自行判断。

MobileOrg Android 是一个 Org-mode 的 Android 客户端,fork 自已经停止维护的 matburt/mobileorg-android。原始代码写于 2013 年左右,targetSdk 为 17(Android 4.2),使用 ActionBarSherlock、Support Library v4 等早已废弃的库。

Google Play 要求 targetSdk 33+,所以必须迁移。本文记录了整个迁移过程中的踩坑和修复经验。

迁移时间线

整个迁移分三个阶段,共 50+ 个 commit:

第一阶段:构建系统升级

原始项目使用 ADT 构建,没有 Gradle 支持。首先需要让项目能编译:

  1. 创建 Gradle 构建配置(AGP 3.0.1 + Gradle 4.1)
  2. 解决 ActionBarSherlock 兼容性(禁用 AAPT2)
  3. 解决 support-v4 版本冲突(降级到 25.4.0)

第二阶段:AndroidX 迁移

  1. 替换 ActionBarSherlock 为 AppCompat
  2. 执行 AndroidX 迁移(Refactor → Migrate to AndroidX)
  3. 升级到现代工具链(AGP 8.2.2 + Gradle 8.5 + JDK 17 + compileSdk 34)

第三阶段:运行时兼容性修复

这是最耗时的阶段。代码能编译,但在现代 Android 上各种崩溃和功能异常。下面详细记录。

踩过的坑

坑 1:ActionBarSherlock → AppCompat 的菜单图标消失

症状:菜单图标在 ActionBar 上不显示

AppCompat 使用自己的 namespace 解析 showAsAction 属性。如果用 =android:showAsAction=,AppCompat 会静默忽略,图标不显示但不报错。

<!-- 错误:AppCompat 会忽略 -->
<item android:showAsAction="ifRoom" />

<!-- 正确:必须用 app namespace -->
<item app:showAsAction="ifRoom"
    xmlns:app="http://schemas.android.com/apk/res-auto" />

**影响范围**:项目中 8 个菜单 XML 文件全部需要修改。只修了一个文件时,其他界面(capture 保存按钮、节点操作按钮等)的图标依然消失。

教训:迁移框架后要全局搜索同类问题,不要只修一个文件就以为搞定了。

坑 2:NotificationChannel 导致前台服务崩溃

症状:CannotPostForegroundServiceNotificationException: Bad notification for startForeground

Android 8.0(API 26)引入了 NotificationChannel。Android 13(API 33)进一步强化——如果 notification 引用的 channel 不存在,直接抛异常崩溃。

原始代码通过反射创建 channel(因为当时 compileSdk 是 23),但反射异常被静默吞掉了。Channel 没创建成功,notification 引用了一个不存在的 channel,崩溃。

// 原来的反射方式——异常被吞掉,channel 创建失败但不知道
try {
    Class<?> channelClass = Class.forName("android.app.NotificationChannel");
    // ...
} catch (Exception e) {
    // 静默失败!
}

// 修复:compileSdk 34 已经有直接 API,不需要反射
NotificationChannel channel = new NotificationChannel(channelId, name,
    NotificationManager.IMPORTANCE_LOW);
nm.createNotificationChannel(channel);

同时,所有 NotificationCompat.Builder 构造函数都必须传 CHANNEL_ID:

// 错误:没有 channel ID
new NotificationCompat.Builder(context)

// 正确
new NotificationCompat.Builder(context, CHANNEL_ID)

**影响范围**:6 处 Builder 调用点,涉及 SyncService、TimeclockService、SynchronizerNotification、SynchronizerNotificationCompat。

坑 3:危险权限的运行时检查

症状:CalendarSyncService 启动时 SecurityException 崩溃

Android 6.0(API 23)引入了运行时权限。=READ_CALENDAR= 和 WRITE_CALENDAR 是危险权限,光在 Manifest 声明不够,必须运行时请求。

// 在 Service.onCreate() 中检查权限
if (!hasCalendarPermission()) {
    stopSelf();  // 无权限则优雅退出
    return;
}

**影响范围**:CalendarSyncService 的 onCreate()onStartCommand() 都需要加权限检查。

坑 4:Service early return 导致 onDestroy NPE

症状:修复权限检查后,CalendarSyncService.onDestroy() 抛 NullPointerException

修复坑 3 加了 early return 后,=sharedPreferences= 没有被初始化,但 Android 仍然会调用 onDestroy()=,里面直接调了 =unregisterOnSharedPreferenceChangeListener(sharedPreferences...) 导致 NPE。

// onDestroy 必须做空检查
@Override
public void onDestroy() {
    if (sharedPreferences != null) {
        sharedPreferences.unregisterOnSharedPreferenceChangeListener(this);
    }
    super.onDestroy();
}

教训:在 onCreate() / onStartCommand() 中加 early return(比如权限检查失败后 stopSelf()=)时,=onDestroy() 仍会被 Android 调用。所有在 onDestroy() 中使用的字段都必须做空检查。

坑 5:主线程网络操作

症状:SSH Connection failed: android.os.NetworkOnMainThreadException

Android 3.0 开始禁止主线程网络操作。但 SyncService.getSynchronizer() 在主线程调用 new SSHSynchronizer()=,而 SSHSynchronizer 的构造函数调了 =connect() 做网络连接。

// 错误:构造函数中发起网络连接
public SSHSynchronizer(Context context) {
    // ... 读取配置 ...
    this.connect();  // 网络操作在主线程!
}

// 修复:移除构造函数中的 connect(),让后台线程处理
public SSHSynchronizer(Context context) {
    // ... 只读取配置,不连接 ...
}

后台线程的 getRemoteFile() 已经有重连逻辑,会自动在正确的线程上建立连接。

坑 6:隐式广播不送达

症状:同步后变更计数不刷新,同步动画不显示,但同步本身是成功的

这是最隐蔽的一个问题。=OrgUtils.announceSyncDone()= 通过 sendBroadcast() 发送隐式广播通知 UI 刷新。但 targetSdk 34 的设备上,隐式广播可能不被 RECEIVER_NOT_EXPORTED 的 receiver 接收。

// 错误:隐式广播,targetSdk 34 可能不送达
Intent intent = new Intent(Synchronizer.SYNC_UPDATE);
context.sendBroadcast(intent);

// 修复:设为显式广播
Intent intent = new Intent(Synchronizer.SYNC_UPDATE);
intent.setPackage(context.getPackageName());  // 关键!
context.sendBroadcast(intent);

这个问题的影响面很大:变更计数不刷新、同步动画不显示、编辑后列表不更新——全部因为广播没送达。但不会报任何错误,就是功能静默失效。

教训:targetSdk 升级后,最危险的不是崩溃,而是功能静默失效。崩溃至少有日志,静默失效只能靠用户反馈。

坑 7:Preference 的隐式 Intent 解析错误

症状:点击 Setup Wizard 直接退回到大纲界面

XML Preference 中用隐式 intent 打开 Activity:

<!-- 错误:隐式 intent 可能解析到错误的 Activity -->
<Preference android:title="Setup Wizard">
    <intent android:action="com.matburt.mobileorg.Settings.SETUP_WIZARD" />
</Preference>

<!-- 修复:显式指定目标 -->
<Preference android:title="Setup Wizard">
    <intent
        android:targetPackage="com.matburt.mobileorg"
        android:targetClass="com.matburt.mobileorg.Gui.Wizard.WizardActivity" />
</Preference>

坑 8:动画在未挂载的 View 上启动

症状:同步图标的旋转动画不显示

startAnimation() 再 =setActionView()=,此时 view 还没挂载到 window,动画被静默忽略。

// 错误顺序
refreshView.startAnimation(rotate);  // view 还没挂载!
synchronizerMenuItem.setActionView(refreshView);

// 修复:先挂载,再用 post 延迟启动动画
synchronizerMenuItem.setActionView(refreshView);
refreshView.post(() -> refreshView.startAnimation(rotate));

坑 9:Intent.getAction() 返回 null 导致 NPE

症状:Setup Wizard 完成后,OutlineActivity 崩溃 NullPointerException

之前修改 Wizard 完成后的导航逻辑,使用 FLAG_ACTIVITY_SINGLE_TOP 回到已有的 OutlineActivity:

Intent outlineIntent = new Intent(context, OutlineActivity.class);
outlineIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);
context.startActivity(outlineIntent);

由于 FLAG_ACTIVITY_SINGLE_TOP=,Android 不会创建新实例,而是调用已有实例的 =onNewIntent()=。但这个 Intent 没有设 action,所以 =getAction() 返回 null:

// 崩溃:intent.getAction() 为 null,对 null 调 equals()
if (intent.getAction().equals(SYNC_FAILED)) { ... }

// 修复:常量在左,天然防 null
if (SYNC_FAILED.equals(intent.getAction())) { ... }

教训:=Intent.getAction()= 可以为 null。使用 FLAG_ACTIVITY_SINGLE_TOP 导航回已有 Activity 时,=onNewIntent()= 收到的 Intent 可能没有 action。养成 CONSTANT.equals(variable) 的习惯可以避免此类 NPE。

经验总结

1. 静默失效比崩溃更可怕

Android 兼容性的最大敌人不是 FATAL EXCEPTION=,而是功能静默消失。=android:showAsAction 被忽略、隐式广播不送达、动画不播放——这些都不会出现在 logcat 里,只能靠人工测试发现。

2. 全局搜索同类问题

修了一个文件的 showAsAction 后,必须全局搜索所有同类问题。Android 迁移中的问题往往是同一模式在多处重复。我们用 grep -r "android:showAsAction" 一次性找到了 7 个文件。

3. 生命周期配对调用

onCreate() 中加 early return 时,必须检查 onDestroy() 是否使用了会被跳过的字段。Android 保证 onDestroy()stopSelf() 后仍被调用。

4. 编译通过≠运行正确

compileSdk 升级后,很多废弃 API 仍然编译通过但运行时行为改变。需要逐个在目标设备上测试。

5. 用 CI 构建而非本地构建

全程通过 GitHub Actions 构建 APK,在真机上测试。本地环境和 CI 环境的差异可能导致遗漏问题。

Android 迁移 MobileOrg 兼容性