- 添加 core/proguard-rules.pro 保留关键类方法签名 - 添加 core/src/main/baseline-prof.txt 预编译热点路径 - 引入 androidx.profileinstaller 依赖 - LunarCache 新增 formatLunarDate() 复用缓存数据 - BottomCard 使用 remember 缓存农历描述避免重复计算 - CalendarMonthView 延迟一帧显示 BottomCard,避免首帧阻塞 - README 补充 Baseline Profiles 维护指南与性能监控说明 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
YaYa
基于 Kotlin Multiplatform 与 Compose Multiplatform 的跨平台日历应用,Android 与 iOS 共享同一套 UI 与业务逻辑。
特性
- 流畅的视图切换 —— 月视图、周视图、年视图三种模式,拖拽手势驱动月↔周折叠,弹簧动画自动吸附
- 无限滑动分页 —— 基于
Int.MAX_VALUE的虚拟分页,前后无边界翻页 - 完整中式日历 —— 公历 + 农历 + 二十四节气 + 传统节日,ISO 8601 周起始(周一)
- 个人排班周期 —— 自定义工作/休息循环,与公共节假日独立
- Material 3 设计 —— 动态配色,深色模式
技术栈
- Kotlin 2.3 · Compose Multiplatform 1.11 · Material 3
kotlinx-datetime处理所有日期逻辑tyme4kt提供农历、节气与传统节日sketch渲染 GIF 动画- 双模块:
:shared(UI + 逻辑) ·:androidApp(薄壳) - iOS 入口为
MainViewController.kt,Xcode 工程位于iosApp/
线条小狗表情包来自 https://www.douban.com/group/topic/264788645/?_i=9181692phrDzjR,9241256phrDzjR
快速开始
# 编译 Android debug APK
./gradlew :app:assembleDebug
# 安装到设备
./gradlew :app:installDebug
# 编译 release APK(含 Baseline Profiles)
./gradlew :app:assembleRelease
Baseline Profiles 维护指南
本项目已集成 Baseline Profiles,用于消除冷启动时的 JIT 编译开销。Release APK 构建时会自动将 core/src/main/baseline-prof.txt 打包进 assets/dexopt/baseline.prof。
何时需要更新 baseline-prof.txt
每次发版前,检查以下清单。任一条件命中,就需要更新 profile:
| 检查项 | 是否需要更新 | 说明 |
|---|---|---|
| 新增/修改了首帧渲染的 composable | 必须 | MainActivity → CalendarMonthView 启动路径上的任何 composable |
修改了 DayCell.kt 的方法签名 |
必须 | DayCell 是启动最热点(35 次调用) |
修改了 CalendarMonthPage.kt 的方法签名 |
必须 | 月度网格页面在首帧渲染 |
修改了 CalendarMonthView.kt 的方法签名 |
必须 | 根 composable |
修改了 LunarCache.kt 的计算逻辑 |
建议 | 缓存 miss 时会走 compute() 路径 |
新增/删除了 tyme4kt 的调用 |
建议 | 农历计算是 CPU 密集型 |
| 仅修改 UI 颜色、文字、布局 | 不需要 | 不涉及方法签名变化 |
| 新增设置页、关于页等非首屏页面 | 不需要 | 不在冷启动路径 |
如何更新
-
定位新增/变更的热点方法
通过 logcat 抓取启动时的 JIT 编译日志:
adb shell setprop dalvik.vm.extra-opts -verbose:compiler adb logcat -s "JIT" | grep "plus/rua/project"或抓取带 ART 详细日志的启动 trace:
adb shell am start -W -n plus.rua.project/.MainActivity adb logcat -d | grep -E "JIT|dex2oat|BaselineProfile" -
编辑
core/src/main/baseline-prof.txt新增规则格式:
HSPL<类全路径;-><方法名>(<参数类型签名>)<返回类型签名>例如新增一个 composable
NewWidget:HSPLplus/rua/project/ui/NewWidgetKt;->NewWidget(Landroidx/compose/ui/Modifier;Landroidx/compose/runtime/Composer;I)V常用签名速查:
Kotlin 类型 Profile 签名 IntIFloatFBooleanZStringLjava/lang/String;LocalDateLkotlinx/datetime/LocalDate;ModifierLandroidx/compose/ui/Modifier;ComposerLandroidx/compose/runtime/Composer;Function0<Unit>Lkotlin/jvm/functions/Function0;Function1<T, R>Lkotlin/jvm/functions/Function1;ShiftKind?Lplus/rua/project/ShiftKind;Unit/ 无返回值V -
验证方法名不被混淆
如果新增的方法在
core/proguard-rules.pro中没有保留规则,添加:-keepclassmembers class plus.rua.project.ui.NewWidgetKt { public static void NewWidget(...); } -
编译验证
./gradlew :core:compileDebugKotlin ./gradlew :app:assembleRelease -
确认 profile 打包成功
unzip -l app/build/outputs/apk/release/app-release-unsigned.apk | grep baseline # 应看到 assets/dexopt/baseline.prof
自动化替代方案(推荐后期迁移)
手动维护容易遗漏,长期建议迁移到 Macrobenchmark 自动生成:
- 创建
:macrobenchmark模块 - 编写启动基准测试(自动遍历冷启动路径)
./gradlew :macrobenchmark:connectedBenchmarkAndroidTest- 自动输出
baseline-prof.txt,直接替换即可
参考:Android Baseline Profiles 官方文档
性能监控
项目内置了 LunarCache(农历/节气/节假日缓存)和性能追踪(ComposeTrace.kt)。查看 DEVELOPMENT.md 了解如何使用 Perfetto/Systrace 进行深度性能分析。
Description
Languages
Kotlin
77.3%
Shell
18.7%
PowerShell
4%