至於調用系統分享這個功能就很簡單了,直接上代碼,主要如果項目已經使用了分區的概念,那麼在Android 11手機上面分享就會提示"獲取資源失敗"的情況
//分享文字
public static void shareText(Context context, String text, String title) {
new Handler(Looper.getMainLooper()).post(new Runnable() {
@Override
public void run() {
Intent intent = new Intent(Intent.ACTION_SEND);
intent.setType("text/plain");
intent.putExtra(Intent.EXTRA_TEXT, text);
context.startActivity(Intent.createChooser(intent, title));
}
});}
//分享圖片
public static void shareImage(Context context, Uri uri, String title) {
Intent intent = new Intent(Intent.ACTION_SEND);
intent.setType("image/png");
intent.putExtra(Intent.EXTRA_STREAM, uri);
context.startActivity(Intent.createChooser(intent, title));
}
1.先分析問題原因
* Android 10 中,Google首次引入了分區存儲,將公共區域劃分成了不同的集合,並且在媒體文件和其他文檔之間建立了清楚的分割。經過劃分之後應用不可以隨意訪問外部存儲區中的文件,而只能訪問媒體文件。
* 1.應用私有目錄:存儲應用私有數據,外部存儲應用私有目錄對應Android/data/packagename,內部存儲應用私有目錄對應data/data/packagename;
* 2.共享目錄:存儲其他應用可訪問文件, 包含媒體文件、文檔文件以及其他文件,對應設備DCIM、Pictures、Alarms, Music, Notifications,Podcasts, Ringtones、Movies、Download等目錄。
* Android 11 (API 級別 30) 進一步增強了平臺功能,爲外部存儲中的應用和用戶數據提供了更好的保護。
* 從 Android 11 開始,使用 分區存儲模式 的應用即使擁有 READ_EXTERNAL_STORAGE 權限,也無法再訪問外部存儲中的任何其他應用的 專屬目錄 中的文件,受此影響應用間分享就需要使用通過應用間共享文件適配的方式( FileProvider進行分享),通過FileProvider,就允許第三方應用讀取你的應用所分享的文件,而不會受到分區存儲的限制。
首先分享應用數據給第三方應用需要在用到文件共享,就需要在AndroidManifest.xml中配置FileProvider,但是配置了只代表你可以使用FileProvider存儲了應用間共享文件,但是在並不代表你使用了FileProvider(後邊會講怎麼使用)。
2.如何使用FileProvider
2.1 首先,在項目的AndroidManifest.xml添加相關配置,示例如下:
<!--
1.applicationId: 應用包名;
2.android:exported:要求必須爲false,不需要暴露組件;
3.android:grantUriPermissions:true,表示授予URI臨時訪問權限;
4.android:authorities這個屬性的值,建議寫包名+fileprovider,當然也可以起別的字符串, 但是在設備中不能出現2個及以上的APP使用到同一個authorities屬性值,因爲無法共存;
-->
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="${applicationId}.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_provider_paths" />
</provider>
在res/xml目錄(如果沒有xml目錄,則新建一個)下,添加文件file_provider_paths.xml,內容如下:
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-files-path name="sharedata" path="shareData/"/>
</paths>
external-files-path表示通過 Context.getExternalFilesDir(null) 接口獲取到的目錄下的文件纔可被共享,其他未配置的路徑均不可被分享。同樣的節點可以配置多個,以支持多個不同的子目錄,如下所示:
第二步使用FileProvider接口,需要判斷sdk
將路徑通過FileProvider的接口轉換成content://URI形式,示例如下:
public static void shareBitmap(Context context, Bitmap bitmap, String title) {
Uri shareUri = null;
try {
//只能在android 11上面才能使用FileProvider存儲圖片,在android10上面同樣取不到
if (Build.VERSION.SDK_INT > 29) {
String sharedata = context.getExternalFilesDir(null) + "/sharedata/" + System.currentTimeMillis() + ".jpg";
ShareHelper.save(bitmap, sharedata, Bitmap.CompressFormat.JPEG);
File file = new File(sharedata);
shareUri=getFileProvider(context,file);
} else {
String filePath = ImageUtil.getNewPhotoPath();
ShareHelper.save(bitmap, filePath, Bitmap.CompressFormat.JPEG);
shareUri=Uri.fromFile(new File(filePath));
}
Intent intent = new Intent(Intent.ACTION_SEND);
intent.setType("image/*");
intent.putExtra(Intent.EXTRA_STREAM,shareUri);
context.startActivity(Intent.createChooser(intent, title));
} catch (Exception e) {
e.printStackTrace();
}
}
// 將File 轉化爲 content://URI
public static Uri getFileProvider(Context context, File file) {
// ‘authority’要與`AndroidManifest.xml`中`provider`配置的`authorities`一致,假設你的應用包名爲com.example.app
String authority = context.getPackageName() + ".provider";
Uri contentUri = FileProvider.getUriForFile(context, authority, file);
// 授權給微信訪問路徑
context.grantUriPermission("com.tencent.mm", // 這裏填微信包名
contentUri, Intent.FLAG_GRANT_READ_URI_PERMISSION);
return contentUri;
}
public static final String getDirPath() {
File cameraPhotoPath = AppMain.getApp().getExternalFilesDir("UploadImage");
if (!cameraPhotoPath.exists()){
cameraPhotoPath.mkdirs();
}
return cameraPhotoPath.getAbsolutePath();
}
public static final String getNewPhotoPath() {
return getDirPath() + "/" + System.currentTimeMillis() + ".jpg";
}