Android targetSdk从22到26的坑

1,774 阅读2分钟

因应用市场的要求,需要将targetSdkVersion的版本改变为26+,所以才有了本篇文章的由来

一.权限问题

毋庸置疑,以前targetSdkVersion是22就是懒得处理权限导致的,应了一句话,欠下的总要还的.

举个栗子,以前你想使用系统相机进行拍照.代码会是这样的.

        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(photoFile));
        startActivityForResult(intent, PHOTO_REQUEST_TAKEPHOTO);

升级之后的代码应该是这样的.

        /*
         * 如果是6.0以上才去判断是否需要判断运行时权限,6.0以下不考虑
         */
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            if (ContextCompat.checkSelfPermission(this,
                    Manifest.permission.CAMERA)
                    != PackageManager.PERMISSION_GRANTED) {
                requestPermissions(new String[]{Manifest.permission.CAMERA}, CAMERA_PERMISSION);
                return;
            }
        }else{
           
        }

申请权限的回调

   @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        switch (requestCode) {
            case CAMERA_PERMISSION:
                if (permissions[0].equals(Manifest.permission.CAMERA)) {
                    if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                       //申请成功之后,跳转拍照界面
                    }
                }
                break;
            default:
                break;
        }
    }

二. 应用间共享文件问题.

Android 7.0 行为变更 通过FileProvider在应用间共享文件

所以如果你还是使用刚才那个打开相机的代码,你会发现,就算有了权限,照样crash.所以你还需要以下操作.

1. 如下图所示.在res文件目录下新建一个xml文件目录,并且创建一个xml文件,文件内容是自定义需要共享的文件夹.

2. AndroidManifest.xml 中配置如下代码

        <provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="${applicationId}.provider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/provider_paths" />
        </provider>

3. 打开相机的代码也要做出如下改变.

            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            if (intent.resolveActivity(getPackageManager()) != null) {
                /*
                 * 指定拍照存储路径
                 * 7.0 及其以上使用FileProvider替换'file://'访问
                 */
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                    //这里的BuildConfig,需要是程序包下BuildConfig。
                    intent.putExtra(MediaStore.EXTRA_OUTPUT, FileProvider.getUriForFile(this, BuildConfig.APPLICATION_ID + ".provider", photoFile));
                    intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                } else {
                    intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(photoFile));
                }
                startActivityForResult(intent, PHOTO_REQUEST_TAKEPHOTO);
            }

三. 打开服务的问题.

1. Android8.0之后不能悄悄的打开后台服务喽,不然你的应用同样会crash,所以你需要添加以下代码

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            int NOTIFICATION_ID = (int) (System.currentTimeMillis()%10000);
            NotificationChannel channel = new NotificationChannel("hh","name", NotificationManager.IMPORTANCE_HIGH);
            NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
            manager.createNotificationChannel(channel);
            Notification notification = new Notification.Builder(getApplicationContext(),"hh").build();
            startForeground(NOTIFICATION_ID,notification);
        }

2. Android 8.0 还对特定函数做出了以下变更:

  • 如果针对 Android 8.0 的应用尝试在不允许其创建后台服务的情况下使用 startService() 函数,则该函数将引发一个 IllegalStateException。
  • 新的 Context.startForegroundService() 函数将启动一个前台服务。现在,即使应用在后台运行,系统也允许其调用 Context.startForegroundService()。不过,应用必须在创建服务后的五秒内调用该服务的 startForeground() 函数。

四. 发送广播问题.

你会发现在8.0上发送广播接收不到了,尴尬不,意外不.<< Android8.0 静态receiver接收不到隐式广播>>

加上一行如下代码就没事了

 intent.setPackage(getPackageName());

持续更新中...也许还有很多问题需要去发现和解决..