需求
在文件管理中,增加一个通过蓝牙分享图片的功能

实现
Android蓝牙分享已经有现成的功能实现,直接上代码:

private void blueToothSendFile() {try {Intent localIntent = null;localIntent = new Intent();localIntent.setAction(Intent.ACTION_SEND);File tempfiles = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES),"1.png");if(!tempfiles.exists()){return;}Uri contentUri = FileProvider.getUriForFile(Objects.requireNonNull(getActivity()).getApplicationContext(), "xx.xx.xxx.fileprovider", tempfiles);localIntent.setType("image/*");localIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION |Intent.FLAG_GRANT_WRITE_URI_PERMISSION);localIntent.setPackage("com.android.bluetooth");localIntent.putExtra(Intent.EXTRA_STREAM, contentUri);startActivityForResult(localIntent, 9527);}catch (Exception e){e.printStackTrace();}}

踩坑

这里记录一下里面需要特别注意的几点:

1、FileUriExposedException

在7.0的以上的系统中,尝试传递 file://URI可能会触发FileUriExposedException,原始代码如下:

Uri contentUri = Uri.fromFile(tempfiles);

异常如下:

01-06 11:45:20.798  9486  9486 W System.err: android.os.FileUriExposedException: file:///storage/emulated/0/Pictures/1.png exposed beyond app through ClipData.Item.getUri()
01-06 11:45:20.798  9486  9486 W System.err:    at android.os.StrictMode.onFileUriExposed(StrictMode.java:1978)
01-06 11:45:20.799  9486  9486 W System.err:    at android.net.Uri.checkFileUriExposed(Uri.java:2371)
01-06 11:45:20.799  9486  9486 W System.err:    at android.content.ClipData.prepareToLeaveProcess(ClipData.java:963)
01-06 11:45:20.799  9486  9486 W System.err:    at android.content.Intent.prepareToLeaveProcess(Intent.java:10219)
01-06 11:45:20.799  9486  9486 W System.err:    at android.content.Intent.prepareToLeaveProcess(Intent.java:10204)
01-06 11:45:20.799  9486  9486 W System.err:    at android.app.Instrumentation.execStartActivity(Instrumentation.java:1667)
01-06 11:45:20.799  9486  9486 W System.err:    at android.app.Activity.startActivityForResult(Activity.java:4621)

解决方案就是使用官方推荐的FileProvider
可以加个判断:

if (Build.VERSION.SDK_INT >= 24) {//android 7.0以上uri = FileProvider.getUriForFile(activity, BuildConfig.APPLICATION_ID.concat(".fileProvider"), file);//android:authorities="${applicationId}.fileProvider"} else {uri = Uri.fromFile(file);}
2、FileProvider
1) 声明FIleProvider
2) 编写XML文件
3) 使用FileProvider

1 ) 、AndroidManifest.xml 中声明FileProvider,并在meta-data,里面指向一个xml文件

<providerandroid:name="android.support.v4.content.FileProvider"android:authorities="packagename.fileProvider"//${applicationId}.fileProviderandroid:grantUriPermissions="true"android:exported="false"><meta-dataandroid:name="android.support.FILE_PROVIDER_PATHS"android:resource="@xml/file_paths" />
</provider>

2 )、编写XML文件file_paths.xml

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android"><root-path name="root" path="" /><files-path name="files" path="." /><cache-path name="cache" path="." /><external-path name="external" path="." /><external-files-path name="name1" path="." /><external-cache-path name="name2" path="." />
</paths>

为什么要写这么个xml文件?

使用content://uri替代file://uri,那么,content://的uri如何定义呢?总不能使用文件路径.
所以,需要一个虚拟的路径对文件路径进行映射,需要编写一个xml文件,通过path以及xml节点确定可访问的目录,通过name属性来映射真实的文件路径

节点的定义:

public class FileProvider extends ContentProvider {private static final String[] COLUMNS = {OpenableColumns.DISPLAY_NAME, OpenableColumns.SIZE };private static final StringMETA_DATA_FILE_PROVIDER_PATHS = "android.support.FILE_PROVIDER_PATHS";private static final String TAG_ROOT_PATH = "root-path"; //--设备根目录 /private static final String TAG_FILES_PATH = "files-path"; // --/data/data/<包名>/filesprivate static final String TAG_CACHE_PATH = "cache-path"; // --/data/data/<包名>/cacheprivate static final String TAG_EXTERNAL = "external-path";//--/storage/emulate/0private static final String TAG_EXTERNAL_FILES = "external-files-path";// --/storage/emulate/0/Android/data/<包名>/filesprivate static final String TAG_EXTERNAL_CACHE = "external-cache-path"; ///storage/emulate/0/Android/data/<包名>/cacheprivate static final String TAG_EXTERNAL_MEDIA = "external-media-path";private static final String ATTR_NAME = "name";private static final String ATTR_PATH = "path";

3)、使用FileProvider

Uri contentUri = FileProvider.getUriForFile(Context, "xx.xx.xxx.fileprovider", tempfiles);
3、Permission Denial: that is not exported from UID 1000
01-01 06:37:41.322 W/InstallStaging( 2110): java.lang.SecurityException: Permission Denial: opening provider android.support.v4.content.FileProvider from ProcessRecord{56d9d3c 2110:com.android.packageinstaller/u0a19} (pid=2110, uid=10019) that is not exported from UID 1000
01-01 06:37:41.322 W/InstallStaging( 2110):     at android.os.Parcel.readException(Parcel.java:2005)
01-01 06:37:41.322 W/InstallStaging( 2110):     at android.os.Parcel.readException(Parcel.java:1951)

看打印,not exported,是否可以设置android:exported="true"呢,答案是否定的,会报错。

java.lang.RuntimeException: Unable to get provider android.support.v4.content.FileProvider: java.lang.SecurityException: Provider must not be exported

解决方案有两种:
1)、grantUriPermission

grantUriPermission("com.android.bluetooth",contentUri,Intent.FLAG_GRANT_READ_URI_PERMISSION|Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

2)、addFlags

Intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

添加之后还是报错:

01-01 06:37:41.208 W/ActivityManager(  680): For security reasons, the system cannot issue a Uri permission grant to content://xxxxxxx.fileprovider/download/QQ.apk [user 0]; use startActivityAsCaller() instead

frameworks/base/services/core/java/com/android/server/am/ActivityManagerService.java

int checkGrantUriPermissionLocked(int callingUid, String targetPkg, GrantUri grantUri,final int modeFlags, int lastTargetUid) {...if ((callingAppId == SYSTEM_UID) || (callingAppId == ROOT_UID)) {if ("com.android.settings.files".equals(grantUri.uri.getAuthority())) {// Exempted authority for// 1. cropping user photos and sharing a generated license html//    file in Settings app// 2. sharing a generated license html file in TvSettings app} else {Slog.w(TAG, "For security reasons, the system cannot issue a Uri permission"+ " grant to " + grantUri + "; use startActivityAsCaller() instead");return -1;}}

可以看到,如果是SYSTEM_UID 或者ROOT_UID 的应用,只有com.android.settings.files这个FileProvider才添加权限成功。

android:sharedUserId="android.uid.system"

那么我们如果是普通应该,不是system UID,可不会报错,如果是sharedUserId : system,我们可能需要修改ActivityManagerService.java这里,把自己生命的FileProvider添加进去,比如我的修改:

if ("com.android.settings.files".equals(grantUri.uri.getAuthority())
+          ||grantUri.uri.getAuthority().contains("funtvfileprovider")) {// Exempted authority for// 1. cropping user photos and sharing a generated license html//    file in Settings app// 2. sharing a generated license html file in TvSettings app} else {Slog.w(TAG, "For security reasons, the system cannot issue a Uri permission"+ " grant to " + grantUri + "; use startActivityAsCaller() instead");return -1;}

实战-Android 蓝牙文件分享相关推荐

  1. android 蓝牙opp流程,Android BluetoothProfile之OPP(蓝牙文件分享流程)

    Bluetooth分享图片流程.就用到了Opp这个Profile. 线面简单的阐述一下一个文件分享的流程.向外传输调用的用的是OppService中的客户端相应的方法,接收调用的是服务端相应的方法. ...

  2. android 原生分享文件,Android原生文件分享

    创建分享 创建Intent并指定Action为Intent.ACTION_SEND. val shareIntent = Intent(Intent.ACTION_SEND) 指定需要发送的内容和类型 ...

  3. Android微信文件分享

    最近在做微信分享相关功能,微信官方提供以下5种分享功能(文字类型分享示例.图片类型分享示例.音乐类型分享示例.视频类型分享示例.网页类型分享示例),官网上有相关demo代码,网址:https://op ...

  4. android蓝牙文件传输的实现

    一.android设备蓝牙通信介绍 1.1配对 两个蓝牙设备在建立通信连接之前需要先彼此感知到对方的存在,这一过程就是配对.使用android蓝牙api进行配对分为以下几步: 1.设备A与B均开始蓝牙 ...

  5. android开发 文件分享到应用,Android 实现文件分享功能(共享多个文件)

    效果如图: 神一样的代码: 针对image代码如下: Intentshare=newIntent(Intent.ACTION_SEND); share.putExtra(Intent.EXTRA_ST ...

  6. android+蓝牙+文件传输,蓝牙文件传输Android

    我在通过蓝牙套接字发送大文件时遇到问题.较小的文件正确传输.我相信正确传输高达161280字节. 编辑:我做了一些测试并缩小了原因.看起来 outStream.write(mybytearray, 0 ...

  7. android开发 文件分享到应用,Android开发之——7.0适配之应用之间共享文件(FileProvider)...

    前言 Android 7.0强制启用了被称作StrictMode的策略,带来的影响就是你的App对外无法暴露file://类型的URI了. 如果你使用Intent携带这样的URI去打开外部App(比如 ...

  8. android蓝牙无法输入密码,由于加密问题,Android蓝牙文件无法写入描述符

    我正在尝试确定使用蓝牙低功耗与支持BLE的设备进行通信的应用程序出了什么问题.令人抓狂的部分是,问题只出现在某些设备上,例如来自Europa的Motorola Moto G3和来自中国的Samsung ...

  9. android -- 蓝牙 bluetooth (四)OPP文件传输

    原址 在前面android -- 蓝牙 bluetooth (一) 入门文章结尾中提到了会按四个方面来写这系列的文章,前面已写了蓝牙打开和蓝牙搜索,这次一起来看下蓝牙文件分享的流程,也就是蓝牙应用op ...

最新文章

  1. 因果推断研究获2021诺贝尔经济学奖,图灵奖得主Judea Pearl祝贺并反对
  2. ElementUI中el-upload怎样上传文件并且传递额外参数给Springboot后台进行接收
  3. 迈克尔逊干涉仪的调整与使用实验报告
  4. TIOBE 11月编程语言排行:Java首次跌出前二,Python 势不可挡。
  5. php 连接符.,PHP怎么在数字之间添加连接符
  6. mysql5.7 设置远程访问
  7. Android自定义滑动进度条,Android自定义View实现圆形水波进度条
  8. 电容器在电路中的作用
  9. java内存分配与回收策略、动态对象年龄判断、空间分配担保
  10. 《数字图像处理 第三版》(冈萨雷斯)——第五章 图像复原与重建
  11. Linux驱动开发|音频驱动
  12. Kail linux中无法定位软件包
  13. tiny4412的I2C驱动实现案例(基于MMA7660)自己写的,亲测有效
  14. 人生经典定律[收藏]
  15. C++ - STL标准库
  16. 100代码搞定C语言游戏开发,编程原来如此简单
  17. 【电脑运用及修理】Mozilla Firefox 浏览器
  18. indexOf()使用详解
  19. 《云云众声》第97期:关于云计算 不可不说的大事小情
  20. 內置函數操作筆記-字典

热门文章

  1. 网易视频云一站式托管音视频技术 助推游戏直播新方向
  2. 关于支付宝证书错误 800A138F
  3. 玻璃桥计算机模拟裂痕是什么,玻璃桥模拟碎裂吓人效果,景区还发出道歉信,最装逼的炒作!...
  4. 2022年最新北京机动车签字授权人模拟试题及答案
  5. 第三代酷睿i3处理器_十代酷睿芯片助力!联想ThinkPad 翼14 Slim轻薄本潜能大释放...
  6. 安装mysql nignix_Node.js 蚕食计划(四)—— Express + SQL Server 搭建电影网站
  7. vue3组件篇 Select
  8. css实现九宫格布局的几种方案
  9. 北京内推 | 京东营销与商业化中心招聘NLP算法工程师/实习生
  10. Python爬虫多线程提升数据下载的性能优化