前言
Android 4.4
- 发布ART虚拟机,提供选项可以开启。
- HttpURLConnection 的底层实现改为了OkHttp。
Android 5.0
- ART成为默认虚拟机,完全代替Dalvik虚拟机。
- Context.bindService() 方法需要显式 Intent,如果提供隐式 intent,将引发异常。
Android 6.0
-
增加运行时权限限制
如果你的应用使用到了危险权限,比如在运行时进行检查和请求权限。checkSelfPermission() 方法用于检查权限,requestPermissions() 方法用于请求权限。 -
取消支持Apache HTTP
Android 6.0 版本移除了对 Apache HTTP 相关类库的支持。要继续使用 Apache HTTP API,您必须先在 build.gradle 文件中声明以下编译时依赖项:
android {useLibrary 'org.apache.http.legacy'}
Apache HttpClient 是Apache开源组织提供的一个开源的项目,它是一个简单的HTTP客户端(并不是浏览器),可以发送HTTP请求,接受HTTP响应。
Android 7.0
- Android 7.0 引入一项新的应用签名方案 APK Signature Scheme v2
- Android 框架执行的 StrictMode API 政策禁止在应用外部公开
file:// URI
, 如果一项包含文件URI
的 intent 离开应用,则应用出现故障,并出现FileUriExposedException
异常。
要应用间共享文件,您应发送一项content:// URI
,并授予URI
临时访问权限。进行此授权的最简单方式是使用 FileProvider 类。
- 声明FileProvider
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="app的包名.fileProvider"
android:grantUriPermissions="true"
android:exported="false">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
<!--androidx版本类路径为:androidx.core.content.FileProvider-->
- 编写xml文件,确定可访问的目录
<paths xmlns:android="http://schemas.android.com/apk/res/android">
//代表设备的根目录new File("/");
<root-path name="root" path="." />
//context.getFilesDir()
<files-path name="files" path="." />
//context.getCacheDir()
<cache-path name="cache" path="." />
//Environment.getExternalStorageDirectory()
<external-path name="external" path="." />
//context.getExternalFilesDirs()
<external-files-path name="name" path="path" />
//getExternalCacheDirs()
<external-cache-path name="name" path="path" />
</paths>
- 使用 FileProvider
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
Uri uri = FileProvider.getUriForFile(CameraActivity.this, "app的包名.fileProvider", photoFile);
} else {
Uri uri = Uri.fromFile(photoFile);
}
Android 8.0
-
修改运行时权限错误
在 Android 8.0 之前,如果应用在运行时请求权限并且被授予该权限,系统会错误地将属于同一权限组并且在清单中注册的其他权限也一起授予应用。对于针对 Android 8.0 的应用,系统只会授予应用明确请求的权限。然而,一旦用户为应用授予某个权限,则所有后续对该权限组中权限的请求都将被自动批准。
也就是说,以前你申请了READ_EXTERNAL_STORAGE权限,应用会同时给你授予同权限组的WRITE_EXTERNAL_STORAGE权限。如果Android8.0以上,只会给你授予你请求的READ_EXTERNAL_STORAGE权限。如果需要WRITE_EXTERNAL_STORAGE权限,还要单独申请,不过系统会立即授予,不会提示。 -
修改通知
Android 8.0 对于通知修改了很多,比如通知渠道、通知标志、通知超时、背景颜色。其中比较重要的就是通知渠道,其允许您为要显示的每种通知类型创建用户可自定义的渠道。
这样的好处就是对于某个应用可以把权限分成很多类,用户来控制是否显示哪些类别的通知。而开发者要做的就是必须设置这个渠道id,否则通知可能会失效。 -
悬浮窗
Android 8.0 以上必须使用新的窗口类型(TYPE_APPLICATION_OVERLAY
)才能显示提醒悬浮窗:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
mWindowParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY
}else {
mWindowParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT
}
-
不允许安装未知来源的应用
Android 8.0 去除了“允许未知来源”选项,所以如果我们的 App 有安装 App 的功能(检查更新之类的),那么会无法正常安装。
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/>
private void installAPK(){
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
boolean hasInstallPermission = getPackageManager().canRequestPackageInstalls();
if (hasInstallPermission) {
//安装应用
} else {
//跳转至“安装未知应用”权限界面,引导用户开启权限
Uri selfPackageUri = Uri.parse("package:" + this.getPackageName());
Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES, selfPackageUri);
startActivityForResult(intent, 100);
}
}else {
//安装应用
}
}
//接收“安装未知应用”权限的开启结果
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == 100) {
installAPK();
}
}
-
Only fullscreen opaque activities can request orientation
只有全屏不透明的 activity 才可以设置方向。这应该是个bug,在Android8.0中出现,8.1中被修复。当然,我们的处理办法就是要么去掉设置方向的代码,要么舍弃透明效果。
Android 9.0
- 在9.0中默认情况下启用网络传输层安全协议 (TLS),默认情况下已停用明文支持。也就是不允许使用http请求,要求使用https。解决办法就是添加网络安全配置:
<application android:networkSecurityConfig="@xml/network_security_config">
<network-security-config>
<base-config cleartextTrafficPermitted="true" />
</network-security-config>
<!--或者在AndroidManifest.xml中配置:
android:usesCleartextTraffic="true"
-->
- 移除 Apache HTTP 客户端
在6.0中取消了对Apache HTTP 客户端的支持,Android9.0中直接移除了该库,要使用的话需要添加配置:
<uses-library android:name="org.apache.http.legacy" android:required="false"/>
- 前台服务调用
Android 9.0 要求创建一个前台服务需要请求 FOREGROUND_SERVICE 权限,否则系统会引发 SecurityException。
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
startForegroundService(intentService);
} else {
startService(intentService);
}
- 不能在非 Activity 环境中启动Activity
在9.0 中,不能直接非 Activity 环境中(比如Service,Application)启动 Activity,否则会崩溃报错,解决办法就是加上FLAG_ACTIVITY_NEW_TASK
。
Intent intent = new Intent(this, TestActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
Android 10
-
分区存储
Android10 中默认开启了分区存储,也就是沙盒模式。应用只能看到本应用专有的目录(通过 Context.getExternalFilesDir() 访问)以及特定类型的媒体。
可是我一定需要升级吗? 如果targetSdkVersion已经指定成了29,也没有关系,假如你还不想进行作用域存储的适配,只需要在AndroidManifest.xml中加入如下配置即可:
<manifest ... >
<application android:requestLegacyExternalStorage="true" ...>
...
</application>
</manifest>
分区存储下,访问文件的方法:
具体存储方法可参考:Android 10,作用域存储
-
权限再次升级
从Android10开始普通应用不再允许请求权限 android.permission.READ_PHONE_STATE。而且,无论你的App是否适配过Android Q(既targetSdkVersion是否大于等于29),均无法再获取到设备IMEI等设备信息。
如果Android10以下设备获取设备IMEI等信息,可以配置最大sdk版本:
<uses-permission android:name="android.permission.READ_PHONE_STATE"
android:maxSdkVersion="28"/>
Android 11
分区存储强制执行
Android11 强制执行分区存储,也就是沙盒模式。这次真的没有关闭功能了,离Android11出来也有一段时间了,还是抓紧适配吧。修改电话号码相关权限
Android 11 更改了您的应用在读取电话号码时使用的与电话相关的权限。
具体改了什么呢?其实就是两个API:
- TelecomManager 类中的
getLine1Number()
方法 - TelecomManager 类中的
getMsisdn()
方法
也就是当用到这两个API的时候,原来的READ_PHONE_STATE
权限不管用了,需要READ_PHONE_NUMBERS
权限才行。
- 必须加上v2签名
- 新增5G相关API
- 后台位置访问权限再次限制
关于 Android11 的适配文章 Android11 最全适配指南
文章参考
Android 各版本迭代改动与适配
Android 10适配要点,作用域存储
Android 11新特性,Scoped Storage又有了新花样