Android开发——Android7.0的适配注意点小结

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/SEU_Calvin/article/details/78942903
首先大致说一下Android7.0特性改动的地方:
(1)应用间共享文件限制;
(2)对部分广播的后台限制、网络数据节省开关等;
(3)多窗口的支持;
(4)通知栏支持直接回复消息等。

针对Android7.0的适配总结如下:

1. 应用间共享文件限制

1.1 问题描述

在Android7.0系统上,Android 框架强制执行了 StrictMode API 政策禁止向应用外公开 file:// URI。 如果一项包含文件 file:// URI类型的Intent离开你的应用,即使用Uri.fromFile(file)传递文件路径给第三方应用,会出现FileUriExposedException 异常,如调用系统相机拍照、裁切照片、根据新下载的APK文件直接打开安装界面等

1.2 解决措施

若要在应用间共享文件,可以发送 content:// URI类型的Uri,并授予 URI 临时访问权限。 进行此授权的最简单方式是使用 FileProvider类。

1.3 调用系统相机拍照实例

在Android7.0之前,如果你想调用系统相机拍照可以通过以下代码来进行:

File file = new File(Environment.getExternalStorageDirectory(), "/temp/"+System.currentTimeMillis() + ".jpg");
if (!file.getParentFile().exists())
    file.getParentFile().mkdirs();
Uri imageUri = Uri.fromFile(file);
Intent intent = new Intent();
intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
//将拍取的照片保存到指定URI
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent,CODE);

在Android7.0上使用上述方式调用系统相拍照则会抛出FileUriExposedException异常。我们可以使用FileProvider来解决这一问题。

1. 在manifest清单文件中注册provider

由于FileProvider是继承ContentProvider,属于四大组件之一,需要在AndroidManifest.xml中配置,配置如下:

 <provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="com.demo.fileprovider"
    android:exported="false"
    android:grantUriPermissions="true">
    
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/file_provider_paths"/>
</provider>

其中exported必须为false,为true则会报安全异常;grantUriPermissions为true表示授予 URI 临时访问权限。

扫描二维码关注公众号,回复: 3346895 查看本文章

2. 指定共享的目录

为了指定共享的目录我们需要在资源(res)目录下创建一个xml目录,然后创建一个xml文件,file_provider_paths.xml的内容如下:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <paths>
        <external-path path="" name="myFile"></external-path>
    </paths>
</resources>

不同标签代表的根目录对应如下:

  • <files-path/> Context.getFilesDir()
  • <external-path/> Environment.getExternalStorageDirectory()
  • <cache-path/> getCacheDir()

3. 使用FileProvider

上述准备工作做完之后,现在我们就可以使用FileProvider了。 还是以调用系统相机拍照为例,我们需要将上述拍照代码修改为如下:

File file = new File(Environment.getExternalStorageDirectory(), "/temp/"+hello + ".jpg");
if (!file.getParentFile().exists())
    file.getParentFile().mkdirs();

Uri imageUri = null;
Intent intent = new Intent();
if (Build.VERSION.SDK_INT >= 24){
    //如果是7.0或以上,通过FileProvider获取文件的Uri
   imageUri = FileProvider.getUriForFile
            (context, "com.demo.fileprovider", file);
    //添加这一句表示对目标应用临时授权该Uri所代表的文件
    intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); 
}else {
    imgUri = Uri.fromFile(imgFile);
}

intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
//将拍取的照片保存到指定URI
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent,CODE);
  • 将之前Uri的scheme类型为file的Uri改成了有FileProvider创建一个content类型的Uri
  • 添加了intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)来对目标应用临时授权该Uri所代表的文件

将getUriForFile方法获取的Uri打印出来如下:

content://com.demo.fileprovider/myFile/temp/hello.jpg`  

因为上述指定的path为path="",所以content://com.demo.fileprovider/myFile/代表的真实路径根目录就是:/storage/emulated/0/。 content://com.demo.fileprovider/myFile/temp/hello.jpg代表的真实路径就是:/storage/emulated/0/temp/hello.jpg。


2. 系统广播删除

Android N关闭了三项系统广播:网络状态变更广播、拍照广播以及录像广播

当网络发生变化时所有注册了隐式监听网络变化的app都会被启动。删除这些广播可以显著提升设备性能和用户体验。同理拍照广播和录视频广播。

在Android N平台下即使在Manifest.xml清单文件中注册了 CONNECTIVITY_ACTION广播,在网络发生变化时也不会接收到任何信息。但是正在前台运行的应用如果在主线程中通过Context.registerReceiver()动态注册了CONNECTIVITY_ACTION广播,该应用程序仍然可以接收到该广播。


3. 更严格的Doze模式

在 Android 6.0 中,在手机关屏且静止时,Doze 模式通过推迟 CPU 和网络操作来延长底池寿命。而 Android N 则在手机关屏时就会一定程度限制 CPU 和网络操作,进入 Doze 模式一段时间后进一步限制 WakeLock、Alarm、GPS 和 Wi-Fi 扫描等,应该关注休眠状态下更多功能受限导致的BUG。


4. 分屏幕适配

从Android N开始,添加了分屏功能。用户可以拖动两个应用之间的分割线,放大其中一个应用,同时缩小另一个。通过拖动两个应用的分割线,应用可占屏幕的1/3、 1/2、 2/3。

在清单的 <activity> 或 <application> 节点中设置以下属性,启用或禁用多窗口显示:

android:resizeableActivity=["true" | "false"]

如果该属性设置为 true,Activity 将能以分屏和自由形状模式启动; 如果此属性设置为 false,Activity 将不支持多窗口模式,且用户尝试在多窗口模式下启动 Activity,该 Activity 将全屏显示。

多窗口模式不会更改 Activity 生命周期。只有最近与用户交互过的 Activity 为活动状态,所有其他 Activity 虽然可见,但均处于暂停状态。应用在暂停状态下可能仍需要继续其操作。 例如视频播放应用不要在onPause() 中暂停程序中的视频,而是在 onStop() 中,并在onStart() 中恢复视频播放。 更多的分屏适配信息请访问官方文档

猜你喜欢

转载自blog.csdn.net/SEU_Calvin/article/details/78942903
今日推荐