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 支持。首先需要让项目能编译:
- 创建 Gradle 构建配置(AGP 3.0.1 + Gradle 4.1)
- 解决 ActionBarSherlock 兼容性(禁用 AAPT2)
- 解决 support-v4 版本冲突(降级到 25.4.0)
第二阶段:AndroidX 迁移
- 替换 ActionBarSherlock 为 AppCompat
- 执行 AndroidX 迁移(Refactor → Migrate to AndroidX)
- 升级到现代工具链(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 环境的差异可能导致遗漏问题。