Android适配之版本适配

\color{red}{Android9.0即API 28-适配}

  • 全面屏、刘海屏的适配:
    Android 9 支持最新的全面屏,其中包含为摄像头和扬声器预留空间的屏幕缺口。
    通过 DisplayCutout 类可确定非功能区域的位置和形状,这些区域不应显示内容。
    要确定这些屏幕缺口区域是否存在及其位置,使用 getDisplayCutout()函数。
    LAYOUT_IN_DISPLAY_CUTOUT_MODE_DEFAULT:
    只有当DisplayCutout完全包含在系统栏中时,才允许窗口扩展到DisplayCutout区域。否则,窗口的布局不会与显示剪切区域重叠。
    LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES:
    屏幕短边有cutout,会延伸过去;若cutout在长边,一定不会延伸过去。
    LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER:
    窗口决不允许与显示剪切区域重叠。
    设置代码:
WindowManager.LayoutParams lp = getWindow().getAttributes();
lp.layoutInDisplayCutoutMode = WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_DEFAULT;
getWindow().setAttributes(lp);
  • 通知功能的变更
    Android 8.0 引入了通知渠道,允许您为要显示的每种通知类型创建可由用户自定义的渠道。 Android 9 通过下列变更简化通知渠道设置:
    1.)屏蔽渠道组:现在,用户可以针对某个应用在通知设置中屏蔽整个渠道组。 您可以使用 isBlocked() 函数确定何时屏蔽一个渠道组,从而不会向该组中的渠道发送任何通知。
    此外,您的应用可以使用全新的 getNotificationChannelGroup() 函数查询当前渠道组设置。
    2.)全新的广播 Intent 类型:现在,当通知渠道和渠道组的屏蔽状态发生变更时,Android 系统将发送广播 Intent。 拥有已屏蔽的渠道或渠道组的应用可以侦听这些 Intent 并做出相应的回应。
    有关这些 Intent 操作和 extra 的更多信息,请参阅 NotificationManager 参考中更新的常量列表。 有关响应广播 Intent 的信息,请参阅广播。

  • 权限收紧
    1.)为了增强用户隐私,Android 9 引入了若干行为变更,如限制后台应用访问设备传感器、限制通过 Wi-Fi 扫描检索到的信息,以及与通话、手机状态和 Wi-Fi 扫描相关的新权限规则和权限组。
    无论采用哪一种目标 SDK 版本,这些变更都会影响运行于 Android 9 上的所有应用。
    2.)Android 9 限制后台应用访问用户输入和传感器数据的能力。
    如果您的应用在运行 Android 9 设备的后台运行,系统将对您的应用采取以下限制:
    您的应用不能访问麦克风或摄像头。
    使用连续报告模式的传感器(例如加速度计和陀螺仪)不会接收事件。
    使用变化或一次性报告模式的传感器不会接收事件。
    如果您的应用需要在运行 Android 9 的设备上检测传感器事件,请使用前台服务。

  • 对使用非 SDK 接口的限制
    目前处理方式扫描代码,避免使用或者使用类似的替代方法,如果避免不了,最粗暴的是try-catch,正统做法是每处都做if-else的Android9适配,如果是第三方库,可以屏蔽入口,或者反编译sdk处理,
    处理起来比较复杂,官方受限制名单也还没完全确定,等待后续解决办法吧。

\color{red}{Android8.0即API 26-27适配}

  • 自定义icon-launcher图标

  • 通知相关修改
    1.)通知渠道组 NotificationChannel,渠道ID channelId的规则,大改很多,包括提供了应用桌面图标有通知显示小圆点,长按图标还能进行消息操作等
    显示小圆点:
    channel.setShowBadge(true);
    右上角还有一个数字角标:
    Notification.Builder builder=new Notification.Builder(this, channelId);
    builder.setNumber(10);
    通知超时:
    Notification.Builder builder=new Notification.Builder(this, channelId);
    builder.setTimeoutAfter(timeout*1000)
    背景颜色:这个属性要生效,必须是持续任务,且是前台服务
    builder.setColorized(true);
    builder.setColor(Color.BLACK);
    通知清除回调:
    系统现在可区分通知是由用户清除,还是由应用自己移除。要查看清除通知的方式,应实现NotificationListenerService类的新onNotificationRemoved()方法
    AndroidManifest里面不要忘记加配置,这么做,我还是可以保证你没有任何回调会发生,因为这里面有个别别窍,要手动开启通知访问权限。来看模拟器上的操作流程,设置>应用和通知>高级>特殊应用权限->通知使用权
    通过判断第三个参数reason是REASON_CANCEL还是REASON_LISTENER_CANCEL就可以知道是用户删除还是系统删除了

  • 后台服务限制
    在后台中运行的服务会消耗系统资源,这可能降低用户体验。 为了缓解这一问题,系统对这些服务施加了一些限制。那么什么情况下应用被视为处于前台?
    1.)具有可见Activity(不管该Activity已启动还是已暂停)
    2.)具有前台服务
    3.)另一个前台应用已关联到该应用(不管是通过绑定到其中一个服务,还是通过使用其中一个内容提供程序)。
    例如,如果另一个应用绑定到该应用的服务,那么该应用处于前台:输入法、壁纸服务、语音等
    如果以上条件均不满足,应用将被视为处于后台。
    startForegroundService方法调用后,记得5秒钟内调用startForeground方法,否则系统将停止服务并声明此应用为ANR。具体使用看官方文档即可,不做赘述。
    4.)后台位置限制:
    之前说的后台服务限制目前只针对目标SDK版本为8.0的app,但是涉及到后台位置限制就是所有版本SDK都要调整的了,它限制后台应用每小时只接收几次位置更新。
    当然,解决的办法是一样的,依然是建立前台服务

  • Android 8.0去除“允许未知来源”选项,需手动确认。
    如果我们的App具备安装App的功能,那么AndroidManifest文件需要包含REQUEST_INSTALL_PACKAGES权限,未声明此权限的应用将无法安装其他应用。
    我们可以选择使用 Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES这个action将用户引导至安装未知应用权限界面。
    同时也可以使用 packageManager.canRequestPackageInstalls()查询此权限的状态
    不过最简单的办法就是直接在AndroidManifest中配置一下就行了,这样会在App调用安装界面的同时,系统会自动询问用户完成授权,体验尚可。

  • 悬浮窗相关
    这个东西篇幅较长,使用频率不高,后面可能会单独一篇文章说这个,在此便不做赘述,有兴趣的可自行了解。

\color{red}{Android7.0即API 24-25适配}

  • FileProvider
    对于app目标版本大于等于24的应用,禁止在应用外部公开 file:// URI , 如果一项包含文件 URI 的 intent 离开应用,则应用出现 FileUriExposedException 异常。
    要应用间共享文件,您应发送一项 content:// URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider类。如需了解有关权限和共享文件的详细信息,请参阅官方文档。
    FileProvider 实际上是 ContentProvider 的一个子类,它的作用也比较明显,file://Uri 不给用,那么换个 Uri 为 content:// 来替代。

接下来看下如何使用:
1、定义 FileProvider
我们先在 AndroidManifest 中进行注册

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

2、指定可分享的文件路径
FileProvider 只能为指定的目录中的文件生成内容 URI。要指定目录,就必须使用 <paths> 元素的子元素在 XML 中指定其存储区域和路径。

我们先创建一个名为 res/xml/file_paths_public.xml 的新文件

<paths>
    <!--图片-->
    <external-path
        name="my_images"
        path="Pictures/" />

    <!--自定义根目录-->
    <root-path
        name="root_path"
        path="." />
</paths>

在 paths 节点内部支持以下几个子节点,分别为:

  • root-path:设备的根目录 new File("/")
  • files-path:context.getFileDir()
  • cache-path:context.getCacheDir()
  • external-path:Environment.getExternalStorageDirectory()
  • external-files-path:context.getExternalFilesDirs()
  • external-cache-path:getExternalCacheDirs()

代码中的常见用法:

Intent intent = new Intent("android.intent.action.VIEW");
intent.addCategory("android.intent.category.DEFAULT");
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); // FileProvider方式必须加这个临时权限
Uri uri = FileProvider.getUriForFile(mContext, "com.hccf.fileprovider", file);
intent.setDataAndType(uri, "application/pdf");
mContext.startActivity(intent);

还有一种用法是: 使用Context 的 grantUriPermission() 方法和revokeUriPermission(); 这种比较少用。

至此,FileProvider的介绍就差不多结束了。

  • V2签名
    APK signature scheme v2:
    Android 7.0 引入一项新的应用签名方案 APK Signature Scheme v2,它能提供更快的应用安装时间和更多针对未授权 APK 文件更改的保护。在默认情况下,Android Studio 2.2 和 Android Plugin for Gradle 2.2 会使用 APK Signature Scheme v2 和传统签名方案来签署您的应用。

使用Android Studio 自带的打包工具的话,只有勾选了了v2签名的方式,app才能在目标版本24及以上的7.0系统手机安装。

还有一些其他的比较少遇到的,就不做赘述了,参考文章:
https://blog.csdn.net/qq_17766199/article/details/77404712

\color{red}{Android6.0即API 23适配}

            // 获取权限
            RxPermissions rxPermissions = new RxPermissions(getActivity());
            rxPermissions.request(Manifest.permission.READ_EXTERNAL_STORAGE,
                    Manifest.permission.WRITE_EXTERNAL_STORAGE)
                    .subscribe(new Observer<Boolean>() {
                        @Override
                        public void onSubscribe(Disposable d) {

                        }

                        @Override
                        public void onNext(Boolean aBoolean) {
                            if (aBoolean){
                                // 权限获取成功 开始下载
                            }
                        }

                        @Override
                        public void onError(Throwable e) {
                            // 获取权限失败,跳转设置页面
                            new PermissionPageUtils(getActivity()).jumpPermissionPage();
                            ToastUtils.showLongBottom("获取权限失败,请检查是否开启所需权限");
                        }

                        @Override
                        public void onComplete() {

                        }
                    });

\color{red}{Android5.0即API 21-22适配}

  • 未完待续


 
转载链接:https://www.jianshu.com/p/db0f8deecd8b
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android系统从5.0开始引入了Material Design风格,其中一个重要的改变是状态栏的颜色和样式。在之前的版本中,状态栏通常是黑色的,而在Material Design中,状态栏可以根据应用程序的主题颜色进行定制化。因此,我们需要在不同的Android版本状态栏。 在Android 5.0及以上版本中,我们可以使用以下代码来实现状态栏颜色的定制: ```java if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { Window window = getWindow(); window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS); window.setStatusBarColor(ContextCompat.getColor(this, R.color.status_bar_color)); } ``` 在Android 4.4及以下版本中,我们可以通过设置一个透明的状态栏来模拟状态栏颜色: ```java if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) { Window window = getWindow(); window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS); ViewGroup decorView = (ViewGroup) window.getDecorView(); int statusBarHeight = getStatusBarHeight(); View statusBarView = new View(this); statusBarView.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, statusBarHeight)); statusBarView.setBackgroundColor(ContextCompat.getColor(this, R.color.status_bar_color)); decorView.addView(statusBarView); ViewGroup rootView = (ViewGroup) ((ViewGroup) findViewById(android.R.id.content)).getChildAt(0); rootView.setFitsSystemWindows(true); rootView.setClipToPadding(true); } ``` 其中,`getStatusBarHeight()`方法可以获取状态栏的高度。 在Android 6.0及以上版本中,状态栏的颜色还可以与应用程序的主题颜色进行匹。我们可以在`styles.xml`中定义一个`colorPrimaryDark`属性来指定状态栏颜色: ```xml <item name="colorPrimaryDark">@color/status_bar_color</item> ``` 在代码中,我们可以通过以下方式来设置主题: ```java if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { getWindow().setStatusBarColor(ContextCompat.getColor(this, R.color.colorPrimaryDark)); } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值