欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android10填坑適配指南(實(shí)際經(jīng)驗(yàn)代碼)

 更新時間:2019年11月26日 10:09:29   作者:黃海彬  
這篇文章主要介紹了Android10填坑適配指南(實(shí)際經(jīng)驗(yàn)代碼),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

今天看到一篇好的文章,分享給大家,膜拜大佬。

Android10填坑適配指南,包含實(shí)際經(jīng)驗(yàn)代碼,絕不照搬翻譯文檔

1.Region.Op相關(guān)異常:java.lang.IllegalArgumentException: Invalid Region.Op - only INTERSECT and DIFFERENCE are allowed

當(dāng) targetSdkVersion >= Build.VERSION_CODES.P 時調(diào)用 canvas.clipPath(path, Region.Op.XXX); 引起的異常,參考源碼如下:

@Deprecated
public boolean clipPath(@NonNull Path path, @NonNull Region.Op op) {
  checkValidClipOp(op);
  return nClipPath(mNativeCanvasWrapper, path.readOnlyNI(), op.nativeInt);
}

private static void checkValidClipOp(@NonNull Region.Op op) {
  if (sCompatiblityVersion >= Build.VERSION_CODES.P
   && op != Region.Op.INTERSECT && op != Region.Op.DIFFERENCE) {
   throw new IllegalArgumentException(
     "Invalid Region.Op - only INTERSECT and DIFFERENCE are allowed");
  }
}

我們可以看到當(dāng)目標(biāo)版本從Android P開始,Canvas.clipPath(@NonNull Path path, @NonNull Region.Op op) ; 已經(jīng)被廢棄,而且是包含異常風(fēng)險的廢棄API,只有 Region.Op.INTERSECT 和 Region.Op.DIFFERENCE 得到兼容,幾乎所有的博客解決方案都是如下簡單粗暴:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
 canvas.clipPath(path);
} else {
 canvas.clipPath(path, Region.Op.XOR);// REPLACE、UNION 等
}

但我們一定需要一些高級邏輯運(yùn)算效果怎么辦?如小說的仿真翻頁閱讀效果,解決方案如下,用Path.op代替,先運(yùn)算Path,再

給canvas.clipPath:
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.P){
 Path mPathXOR = new Path();
 mPathXOR.moveTo(0,0);
 mPathXOR.lineTo(getWidth(),0);
 mPathXOR.lineTo(getWidth(),getHeight());
 mPathXOR.lineTo(0,getHeight());
 mPathXOR.close();
 //以上根據(jù)實(shí)際的Canvas或View的大小,畫出相同大小的Path即可
 mPathXOR.op(mPath0, Path.Op.XOR);
 canvas.clipPath(mPathXOR);
}else {
 canvas.clipPath(mPath0, Region.Op.XOR);
}

2.明文HTTP限制

當(dāng) targetSdkVersion >= Build.VERSION_CODES.P 時,默認(rèn)限制了HTTP請求,并出現(xiàn)相關(guān)日志:

java.net.UnknownServiceException: CLEARTEXT communication to xxx not permitted by network security policy

第一種解決方案:在AndroidManifest.xml中Application添加如下節(jié)點(diǎn)代碼

<application android:usesCleartextTraffic="true">

第二種解決方案:在res目錄新建xml目錄,已建的跳過 在xml目錄新建一個xml文件network_security_config.xml,然后在AndroidManifest.xml中Application添加如下節(jié)點(diǎn)代碼

android:networkSecurityConfig="@xml/network_config"

名字隨機(jī),內(nèi)容如下:

<?xml version="1.0" encoding="utf-8"?>
<network-security-config>
 <base-config cleartextTrafficPermitted="true" />
</network-security-config>

3.Android Q中的媒體資源讀寫

1、掃描系統(tǒng)相冊、視頻等,圖片、視頻選擇器都是通過ContentResolver來提供,主要代碼如下:

private static final String[] IMAGE_PROJECTION = {
   MediaStore.Images.Media.DATA,
   MediaStore.Images.Media.DISPLAY_NAME,
   MediaStore.Images.Media._ID,
   MediaStore.Images.Media.BUCKET_ID,
   MediaStore.Images.Media.BUCKET_DISPLAY_NAME};

 Cursor imageCursor = mContext.getContentResolver().query(
     MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
     IMAGE_PROJECTION, null, null, IMAGE_PROJECTION[0] + " DESC");

String path = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[0]));
String name = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[1]));
int id = imageCursor.getInt(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[2]));
String folderPath = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[3]));
String folderName = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[4]));

//Android Q 公有目錄只能通過Content Uri + id的方式訪問,以前的File路徑全部無效,如果是Video,記得換成MediaStore.Videos
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q){
  path = MediaStore.Images.Media
      .EXTERNAL_CONTENT_URI
      .buildUpon()
      .appendPath(String.valueOf(id)).build().toString();
 }

2、判斷公有目錄文件是否存在,自Android Q開始,公有目錄File API都失效,不能直接通過new File(path).exists();判斷公有目錄文件是否存在,正確方式如下:

public static boolean isAndroidQFileExists(Context context, String path){
  AssetFileDescriptor afd = null;
  ContentResolver cr = context.getContentResolver();
  try {
   Uri uri = Uri.parse(path);
   afd = cr.openAssetFileDescriptor(uri, "r");
   if (afd == null) {
    return false;
   } else {
    close(afd);
   }
  } catch (FileNotFoundException e) {
   return false;
  }finally {
   close(afd);
  }
  return true;
}

3、copy或者下載文件到公有目錄,保存Bitmap同理,如Download,MIME_TYPE類型可以自行參考對應(yīng)的文件類型,這里只對APK作出說明,從私有目錄copy到公有目錄demo如下(遠(yuǎn)程下載同理,只要拿到OutputStream即可,亦可下載到私有目錄再copy到公有目錄):

public static void copyToDownloadAndroidQ(Context context, String sourcePath, String fileName, String saveDirName){
  ContentValues values = new ContentValues();
  values.put(MediaStore.Downloads.DISPLAY_NAME, fileName);
  values.put(MediaStore.Downloads.MIME_TYPE, "application/vnd.android.package-archive");
  values.put(MediaStore.Downloads.RELATIVE_PATH, "Download/" + saveDirName.replaceAll("/","") + "/");

  Uri external = MediaStore.Downloads.EXTERNAL_CONTENT_URI;
  ContentResolver resolver = context.getContentResolver();

  Uri insertUri = resolver.insert(external, values);
  if(insertUri == null) {
   return;
  }

  String mFilePath = insertUri.toString();

  InputStream is = null;
  OutputStream os = null;
  try {
   os = resolver.openOutputStream(insertUri);
   if(os == null){
    return;
   }
   int read;
   File sourceFile = new File(sourcePath);
   if (sourceFile.exists()) { // 文件存在時
    is = new FileInputStream(sourceFile); // 讀入原文件
    byte[] buffer = new byte[1444];
    while ((read = is.read(buffer)) != -1) {
     os.write(buffer, 0, read);
    }
   }
  } catch (Exception e) {
   e.printStackTrace();
  }finally {
   close(is,os);
  }

}

4、保存圖片相關(guān)

 /**
  * 通過MediaStore保存,兼容AndroidQ,保存成功自動添加到相冊數(shù)據(jù)庫,無需再發(fā)送廣播告訴系統(tǒng)插入相冊
  *
  * @param context  context
  * @param sourceFile 源文件
  * @param saveFileName 保存的文件名
  * @param saveDirName picture子目錄
  * @return 成功或者失敗
  */
 public static boolean saveImageWithAndroidQ(Context context,
             File sourceFile,
             String saveFileName,
             String saveDirName) {
  String extension = BitmapUtil.getExtension(sourceFile.getAbsolutePath());

  ContentValues values = new ContentValues();
  values.put(MediaStore.Images.Media.DESCRIPTION, "This is an image");
  values.put(MediaStore.Images.Media.DISPLAY_NAME, saveFileName);
  values.put(MediaStore.Images.Media.MIME_TYPE, "image/png");
  values.put(MediaStore.Images.Media.TITLE, "Image.png");
  values.put(MediaStore.Images.Media.RELATIVE_PATH, "Pictures/" + saveDirName);

  Uri external = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
  ContentResolver resolver = context.getContentResolver();

  Uri insertUri = resolver.insert(external, values);
  BufferedInputStream inputStream = null;
  OutputStream os = null;
  boolean result = false;
  try {
   inputStream = new BufferedInputStream(new FileInputStream(sourceFile));
   if (insertUri != null) {
    os = resolver.openOutputStream(insertUri);
   }
   if (os != null) {
    byte[] buffer = new byte[1024 * 4];
    int len;
    while ((len = inputStream.read(buffer)) != -1) {
     os.write(buffer, 0, len);
    }
    os.flush();
   }
   result = true;
  } catch (IOException e) {
   result = false;
  } finally {
   close(os, inputStream);
  }
  return result;
}

4.EditText默認(rèn)不獲取焦點(diǎn),不自動彈出鍵盤

該問題出現(xiàn)在 targetSdkVersion >= Build.VERSION_CODES.P 情況下,且設(shè)備版本為Android P以上版本,解決方法在onCreate中加入如下代碼,可獲得焦點(diǎn),如需要彈出鍵盤可延遲一下:

mEditText.post(() -> {
  mEditText.requestFocus();
  mEditText.setFocusable(true);
  mEditText.setFocusableInTouchMode(true);
});

5.安裝APK Intent及其它共享文件相關(guān)Intent

/*
* 自Android N開始,是通過FileProvider共享相關(guān)文件,但是Android Q對公有目錄 File API進(jìn)行了限制,只能通過Uri來操作,
* 從代碼上看,又變得和以前低版本一樣了,只是必須加上權(quán)限代碼Intent.FLAG_GRANT_READ_URI_PERMISSION
*/ 
private void installApk() {
  if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q){
   //適配Android Q,注意mFilePath是通過ContentResolver得到的,上述有相關(guān)代碼
   Intent intent = new Intent(Intent.ACTION_VIEW);
   intent.setDataAndType(Uri.parse(mFilePath) ,"application/vnd.android.package-archive");
   intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
   intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
   startActivity(intent);
   return ;
  }

  File file = new File(saveFileName + "demo.apk");
  if (!file.exists())
   return;
  Intent intent = new Intent(Intent.ACTION_VIEW);
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
   intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
   intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
   Uri contentUri = FileProvider.getUriForFile(getApplicationContext(), "net.oschina.app.provider", file);
   intent.setDataAndType(contentUri, "application/vnd.android.package-archive");
  } else {
   intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");
   intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  }
  startActivity(intent);
}

6.Activity透明相關(guān),windowIsTranslucent屬性

Android Q 又一個天坑,如果你要顯示一個半透明的Activity,這在android10之前普通樣式Activity只需要設(shè)置windowIsTranslucent=true即可,但是到了AndroidQ,它沒有效果了,而且如果動態(tài)設(shè)置View.setVisibility(),界面還會出現(xiàn)殘影...

解決辦法:使用Dialog樣式Activity,且設(shè)置windowIsFloating=true,此時問題又來了,如果Activity根布局沒有設(shè)置fitsSystemWindow=true,默認(rèn)是沒有侵入狀態(tài)欄的,使界面看上去正常。

7.剪切板兼容

Android Q中只有當(dāng)應(yīng)用處于可交互情況(默認(rèn)輸入法本身就可交互)才能訪問剪切板和監(jiān)聽剪切板變化,在onResume回調(diào)也無法直接訪問剪切板,這么做的好處是避免了一些應(yīng)用后臺瘋狂監(jiān)聽響應(yīng)剪切板的內(nèi)容,瘋狂彈窗。

因此如果還需要監(jiān)聽剪切板,可以使用應(yīng)用生命周期回調(diào),監(jiān)聽APP后臺返回,延遲幾毫秒訪問剪切板,再保存最后一次訪問得到的剪切板內(nèi)容,每次都比較一下是否有變化,再進(jìn)行下一步操作。

8.第三方分享圖片等操作,直接使用文件路徑的,如QQ圖片分享,都需要注意,這是不可行的,都只能通過MediaStore等API,拿到Uri來操作

這些是我們根據(jù)sdk升級到29時遇到的實(shí)際問題而羅列出來的,不是翻譯AndroidQ中的行為變更,具體問題請根據(jù)自身實(shí)際自行解決。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Android組件必學(xué)之TabHost使用方法詳解

    Android組件必學(xué)之TabHost使用方法詳解

    這篇文章主要為大家詳細(xì)介紹了Android組件中的TabHost組件使用方法,如何利用TabHost定義Tab標(biāo)簽樣式,感興趣的小伙伴們可以參考一下
    2016-05-05
  • android10 隱藏SystemUI鎖屏下的多用戶圖標(biāo)的示例代碼

    android10 隱藏SystemUI鎖屏下的多用戶圖標(biāo)的示例代碼

    這篇文章主要介紹了android10 隱藏SystemUI鎖屏下的多用戶圖標(biāo),本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-12-12
  • Android實(shí)現(xiàn)保持屏幕常亮功能

    Android實(shí)現(xiàn)保持屏幕常亮功能

    本篇文章主要介紹了Android實(shí)現(xiàn)保持屏幕常亮功能,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-11-11
  • Android動態(tài)替換Application實(shí)現(xiàn)

    Android動態(tài)替換Application實(shí)現(xiàn)

    這篇文章主要介紹了Android動態(tài)替換Application實(shí)現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-05-05
  • Android閃屏效果實(shí)現(xiàn)方法

    Android閃屏效果實(shí)現(xiàn)方法

    這篇文章主要介紹了Android閃屏效果實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了Android閃屏效果的實(shí)現(xiàn)原理及相關(guān)功能與布局設(shè)置技巧,需要的朋友可以參考下
    2016-01-01
  • Android?Studio中如何修改APP圖標(biāo)和APP名稱

    Android?Studio中如何修改APP圖標(biāo)和APP名稱

    這篇文章主要介紹了Android?Studio中如何修改APP圖標(biāo)和APP名稱,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-11-11
  • Android框架Volley使用:ImageRequest請求實(shí)現(xiàn)圖片加載

    Android框架Volley使用:ImageRequest請求實(shí)現(xiàn)圖片加載

    這篇文章主要介紹了Android框架Volley使用:ImageRequest請求實(shí)現(xiàn)圖片加載的相關(guān)知識,非常不錯,具有一定的參考借鑒價值 ,需要的朋友可以參考下
    2019-05-05
  • Android仿淘寶訂單頁面效果

    Android仿淘寶訂單頁面效果

    這篇文章主要介紹了Android仿淘寶訂單頁面效果,電商項(xiàng)目的訂單管理模塊,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-12-12
  • Android控件之RatingBar自定義星級評分樣式

    Android控件之RatingBar自定義星級評分樣式

    RatingBar為評分條控件,默認(rèn)效果為若干個綠色的星星,如果想將其換成其他自定義圖片就要自定義它的style。接下來通過本文給大家介紹Android控件之RatingBar自定義星級評分樣式,感興趣的朋友一起學(xué)習(xí)吧
    2016-02-02
  • Android RecyclerView添加頭部和底部的方法

    Android RecyclerView添加頭部和底部的方法

    這篇文章主要為大家詳細(xì)介紹了Android RecyclerView添加頭部和底部的方法,感興趣的小伙伴們可以參考一下
    2016-05-05

最新評論