Android targetSdk從22到26的坑

因應用市場的要求,須要將targetSdkVersion的版本改變爲26+,因此纔有了本篇文章的由來android

一.權限問題

毋庸置疑,之前targetSdkVersion是22就是懶得處理權限致使的,應了一句話,欠下的總要還的.bash

舉個栗子,之前你想使用系統相機進行拍照.代碼會是這樣的.

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.因此你還須要如下操做.app

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接收不到隱式廣播>>ide

加上一行以下代碼就沒事了

intent.setPackage(getPackageName());
複製代碼

持續更新中...也許還有不少問題須要去發現和解決..函數

相關文章
相關標籤/搜索