实战-Android 蓝牙文件分享
需求
在文件管理中,增加一个通过蓝牙分享图片的功能
实现
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 蓝牙文件分享相关推荐
- android 蓝牙opp流程,Android BluetoothProfile之OPP(蓝牙文件分享流程)
Bluetooth分享图片流程.就用到了Opp这个Profile. 线面简单的阐述一下一个文件分享的流程.向外传输调用的用的是OppService中的客户端相应的方法,接收调用的是服务端相应的方法. ...
- android 原生分享文件,Android原生文件分享
创建分享 创建Intent并指定Action为Intent.ACTION_SEND. val shareIntent = Intent(Intent.ACTION_SEND) 指定需要发送的内容和类型 ...
- Android微信文件分享
最近在做微信分享相关功能,微信官方提供以下5种分享功能(文字类型分享示例.图片类型分享示例.音乐类型分享示例.视频类型分享示例.网页类型分享示例),官网上有相关demo代码,网址:https://op ...
- android蓝牙文件传输的实现
一.android设备蓝牙通信介绍 1.1配对 两个蓝牙设备在建立通信连接之前需要先彼此感知到对方的存在,这一过程就是配对.使用android蓝牙api进行配对分为以下几步: 1.设备A与B均开始蓝牙 ...
- android开发 文件分享到应用,Android 实现文件分享功能(共享多个文件)
效果如图: 神一样的代码: 针对image代码如下: Intentshare=newIntent(Intent.ACTION_SEND); share.putExtra(Intent.EXTRA_ST ...
- android+蓝牙+文件传输,蓝牙文件传输Android
我在通过蓝牙套接字发送大文件时遇到问题.较小的文件正确传输.我相信正确传输高达161280字节. 编辑:我做了一些测试并缩小了原因.看起来 outStream.write(mybytearray, 0 ...
- android开发 文件分享到应用,Android开发之——7.0适配之应用之间共享文件(FileProvider)...
前言 Android 7.0强制启用了被称作StrictMode的策略,带来的影响就是你的App对外无法暴露file://类型的URI了. 如果你使用Intent携带这样的URI去打开外部App(比如 ...
- android蓝牙无法输入密码,由于加密问题,Android蓝牙文件无法写入描述符
我正在尝试确定使用蓝牙低功耗与支持BLE的设备进行通信的应用程序出了什么问题.令人抓狂的部分是,问题只出现在某些设备上,例如来自Europa的Motorola Moto G3和来自中国的Samsung ...
- android -- 蓝牙 bluetooth (四)OPP文件传输
原址 在前面android -- 蓝牙 bluetooth (一) 入门文章结尾中提到了会按四个方面来写这系列的文章,前面已写了蓝牙打开和蓝牙搜索,这次一起来看下蓝牙文件分享的流程,也就是蓝牙应用op ...
最新文章
- 因果推断研究获2021诺贝尔经济学奖,图灵奖得主Judea Pearl祝贺并反对
- ElementUI中el-upload怎样上传文件并且传递额外参数给Springboot后台进行接收
- 迈克尔逊干涉仪的调整与使用实验报告
- TIOBE 11月编程语言排行:Java首次跌出前二,Python 势不可挡。
- php 连接符.,PHP怎么在数字之间添加连接符
- mysql5.7 设置远程访问
- Android自定义滑动进度条,Android自定义View实现圆形水波进度条
- 电容器在电路中的作用
- java内存分配与回收策略、动态对象年龄判断、空间分配担保
- 《数字图像处理 第三版》(冈萨雷斯)——第五章 图像复原与重建
- Linux驱动开发|音频驱动
- Kail linux中无法定位软件包
- tiny4412的I2C驱动实现案例(基于MMA7660)自己写的,亲测有效
- 人生经典定律[收藏]
- C++ - STL标准库
- 100代码搞定C语言游戏开发,编程原来如此简单
- 【电脑运用及修理】Mozilla Firefox 浏览器
- indexOf()使用详解
- 《云云众声》第97期:关于云计算 不可不说的大事小情
- 內置函數操作筆記-字典
热门文章
- 网易视频云一站式托管音视频技术 助推游戏直播新方向
- 关于支付宝证书错误 800A138F
- 玻璃桥计算机模拟裂痕是什么,玻璃桥模拟碎裂吓人效果,景区还发出道歉信,最装逼的炒作!...
- 2022年最新北京机动车签字授权人模拟试题及答案
- 第三代酷睿i3处理器_十代酷睿芯片助力!联想ThinkPad 翼14 Slim轻薄本潜能大释放...
- 安装mysql nignix_Node.js 蚕食计划(四)—— Express + SQL Server 搭建电影网站
- vue3组件篇 Select
- css实现九宫格布局的几种方案
- 北京内推 | 京东营销与商业化中心招聘NLP算法工程师/实习生
- Python爬虫多线程提升数据下载的性能优化