综合
MinSDK - 首选支持MethodHandle的26+。 Android各版本统计(英文版数据较新)
Android 14起Java 17 LTS新特性:HexFormat、record 和 多行文本块儿; Android 14起访问或选取媒体文件时会多出一项“选择照片和视频”(READ_MEDIA_VISUAL_USER_SELECTED权限),避免只能全部允许的风险。 Android 13起后台Toast弹框(前台则正常弹)必须静态和动态申请POST_NOTIFICATIONS权限,故只能换用前台Service?
工具
Android Studio开发IDE官方下载(安装后空间占用10GiB+)
Android Gradle Plugin(AGP)版本列表
aab转apks并部署:参数--connected-device和--adb均为可选 java -jar E:\bundletool-all-1.15.5.jar build-apks --bundle=D:\projects\NewTileMatching.aab --output=D:\projects\NewTileMatching.aab.apks --connected-device java -jar E:\bundletool-all-1.15.5.jar install-apks --apks=D:\projects\NewTileMatching.aab.apks --adb=C:\Users\aw\AppData\Local\Android\Sdk\platform-tools\adb.exe
文章
Android核心服务、组件
谷歌三件套
Google Services Framework(com.google.android.gsf)
Google Play Services(com.google.android.gms)
Google Play Store
发布、测试
Google Play应用市场上架要求 Google Play开发者账号闲置规则 Android反射non-sdk限制分类(unsupported=允许反射) 解除限制:adb shell settings put global hidden_api_policy 1 Android Studio默认提供了两种测试类型package:Instrumented test(插桩测试)和Unit test(单元测试) 注意 - 可通过Firebase Test Lab控制台(“运行测试”按钮下拉选类型)上传apk或命令行、API等方式执行测试; Test Lab控制台插桩测试时(gradle assembleDebugAndroidTest),第一个apk上传含主启动Activity的apk,第二个上传测试apk;否则报:"The uploaded APK does not specify a main launcher activity."。 Test Lab控制台Robo测试时(gradle assembleDebug),若还未给Release版生成签名,可先上传Debug版本apk版解决无签名报错,同时还应上传在AS上录制的Robo脚本(Tools > Firebase -> Test Lab -> Record Robo Script)。 Test Lab控制台测试前,确认下真机版本跟待测app是否兼容。 Instrumented test(插桩测试) - 引入了UI测试库Espresso但未使用; 测试命令 - [只能Debug] gradle connectedStoreDebugAndroidTest 或 connectedFullDebugAndroidTest、connectedAndroidTest、cAT app/build/reports/androidTests/connected/debug/flavors/store/com.wangxiaodong.apps.playphone.ExampleInstrumentedTest.html assembleAndroidTest app\build\outputs\apk\androidTest\store\debug\app-store-debug-androidTest.apk 测试时截图 - https://github.com/android/testing-samples/blob/main/ui/espresso/ScreenshotSample/app/src/androidTest/java/com/example/android/testing/espresso/screenshotsample/ScreenshotJavaTest.java Google Play测试:开放式测试(暴露页面)、封闭式测试、内部测试(免审核) 注意 - 交由指定测试人员进行人工测试前,会先执行自动化测试,但也能关闭该前置自动化测试(即回放Robo预录脚本的“发布前测试报告”)。 内部测试 - xxx
编程
Android后台线程执行前台UI操作:
// 解决 Can't toast on a thread that has not called Looper.prepare()
28+ Use:getApplicationContext().getMainExecutor().execute(() -> { });
new Handler(Looper.getMainLooper()).post(() -> {//避免用Looper.prepare()
Toast.makeText(context, "...", Toast.LENGTH_SHORT).show(); });
View控件:
根View对象(ViewGroup/含标题栏) - activity.this.getWindow().getDecorView() 等同 view.getRootView();
内容视图(布局文件/不含标题栏) - activity.this.findViewById(android.R.id.content)
Toast前台替代 - Snackbar.make(view.getRootView(), "msg", Snackbar.LENGTH_SHORT).show();
FloatingActionButton(FAB/基于ImageView) - 比普通控件多一个交互动画。
通过名字获取resId:
@SuppressLint("DiscouragedApi") // 允许用但性能差,为0则无此resId。
getResources().getIdentifier("google_app_id", "string", getPackageName());
权限:
静态权限 -
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="com.google.android.gms.permission.AD_ID" />
FOREGROUND_SERVICE(前台Service)、RECEIVE_BOOT_COMPLETED
需要用户明确同意的权限 -
POST_NOTIFICATIONS(通知)
需要Google Play控制台审批的权限 -
USE_FULL_SCREEN_INTENT(闹钟、电话可在Play控制台申请默认启用)
共享存储:
高于maxSdkVersion版本的则废弃此权限:
<uses-permission
android:name="android.permission.READ_EXTERNAL_STORAGE"
android:maxSdkVersion="32" />
<uses-permission
android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="29" />
用以下细化权限项来替代:
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />
以及 READ_MEDIA_VIDEO READ_MEDIA_AUDIO
Android 11起采用了包可见性机制,默认可见应用有:adb shell dumpsys package queries
forceQueryable:标签下的apk 表示 可见所有已安装的apk
queries via package name: 标签表示只有配置了包名 才对该app可见
queries via intent: 标签表示只有配置了intent 才对该app可见
queryable via uses-library:
queryable via interaction:
应用包安装:
显式安装权限 - android.permission.REQUEST_INSTALL_PACKAGES
静默安装权限(只能用于系统应用) - android.permission.INSTALL_PACKAGES
WebView载入目录索引页会报JavaScript脚本xxx未定义而页面空白
解决 AlertDialog+getApplicationContext() 报错 Unable to add window -- token null is not valid; 例外:Toast可用任何上下文及getApplicationContext()
new AlertDialog.Builder(view.getContext()).create();// or MainActivity.this
解决横竖屏切换阻止构建 android:screenOrientation="portrait" tools:ignore="LockedOrientationActivity" tools:replace="android:screenOrientation"
Gradle构建:
android {
buildFeatures { buildConfig = true } // 启用后才会生成BuildConfig.DEBUG
defaultConfig {
// x86_64、x86仅用于模拟器调试或ChromeOS系统
ndk.abiFilters 'arm64-v8a', 'x86_64','x86' // 'armeabi-v7a','arm64-v8a','x86','x86_64'
}
buildTypes {
debug { ndk.abiFilters 'arm64-v8a', 'x86_64' }
release { //noinspection ChromeOsAbiSupport
ndk.abiFilters 'arm64-v8a' } // 只保留需要支持的系统so库
// 本地依赖库无名为other的buildType时进行降级
other { matchingFallbacks = ['debug', 'qa', 'release'] }
}
}
AdMob for Android Plugin:
// AdMob task processFullDebugGoogleServices: No matching client found for package name
afterEvaluate {
project.tasks.forEach { task ->
if (task.name.startsWith("processFull")
&& task.name.endsWith("GoogleServices")
) {
println("Skip " + task.name + " task.")
task.onlyIf { return@onlyIf false }
}
}
}
开机解锁前应用执行 - android:directBootAware="true"只是标记下,并不会开机启动,还应标记下注册了LOCKED_BOOT_COMPLETED的receiver。
开机解锁后应用执行 - BOOT_COMPLETED
解决 Duplicate class kotlin... - implementation(platform("org.jetbrains.kotlin:kotlin-bom:1.8.20"))
曾被报道:《尴尬!鸿蒙系统显原形,中文切换到英文,Harmony秒变安卓?》