Android11中相机的变化

Android 11 相机的使用又有了一些变化

距离我的上篇博客间隔快两年了,这两年的时间中学了vue、也学习了小程序和React,对Android搞得又少了一些,所以就没怎么写博客,最近换了份工作,又重新回到了Android岗位,所以又打算写博客,记录一些自己在项目中遇到的一些坑或者新的知识点什么的。


来到新的公司,接手以公司的项目,发现了一个在Android11中使用相机的坑,在11以下的版本中都没有任何问题,只在Android11中出现了。废话也不多说了,下面正文开始

Android11相机获取到的resultCode==0

如果相机正常拍照返回的话resultCode的值是resultCode==-1的,如下图展示
正常返回值
resultCode==0的情况
错误的返回值

简单找了下出现这种情况的原因,出现resultCode==0大概有两种情况

1、运行时权限没有赋予

在Android:targetSdkVersion>23时,就需要在项目中进行申请运行时权限,因为图片的存储需要android.permission.WRITE_SETTINGS的权限,如果没有这种权限的话就会出现resultCode==0的情况。
但是这种情况一般不会出现,除非你是个新手!!!

2、要存储的目录不存在

一般我们在用系统相机进行拍照的时候,都会去创建一个我们自己的目录去存储照片。一般我们都是先指定一个目录路径,去判断这个路径所指向的文件或者文件夹存不存在,如果不存在就进行创建,如果存在则直接使用。
在Android10一下这一套代码都是可以使用的,用起来也是没有问题的。


Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
if (activity == null) {
    return;
}
if (intent.resolveActivity(activity.getPackageManager()) == null) {
    return;
}

File dataDirectoryFile = new File(Environment.getExternalStorageDirectory()+ "/img/");
if (!dataDirectoryFile.exists()) {
	dataDirectoryFile.mkdir();
}
File file = new File(dataDirectoryFile, "fileName.jpg");
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.putExtra(MediaStore.EXTRA_OUTPUT,
    FileProvider.getUriForFile(activity,
        activity.getApplicationContext().getPackageName() + ".provider", file));
activity.startActivityForResult(intent, 10010);

但是上面的代码在Android11的手机上运行的话就会遇到咱们开头所说的问题,resultCode==0的问题。
为了解决问题在百度和谷歌上搜了搜,发现他们只是简单的说了下原因,并没有提出怎么解决。
于是乎我又去谷歌的官方文档上找了找,发现了谷歌官方所推荐的用法并不是这样,于是就按照官方的文档去修改了一下,发现官方果然是没错的!
好的,此文到此结束,请大家参考官方文档进行开发,好的谢谢大家的观看!!!

开个玩笑

主要原因是谷歌在Android10的时候改变的文件的存储策略

1、Android Q文件存储机制修改成了沙盒模式,类似于iOS
2、应用只能访问自己沙盒下的文件和公共媒体文件

到这里思路应该就清晰了,Android11在Android10的基础上对这个文件存储的策略实施的更加严格了,所以在android11的版本上使用原来的创建存储就会出现问题。
所以我就对现有的代码进行了简单的更改

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);// 设置打开相机
if (activity == null) {
    return;
}
	// Ensure that there's a camera activity to handle the intent
	// 官方注释,确保有一个活动来打开相机意图
if (intent.resolveActivity(activity.getPackageManager()) == null) {
    return;
}

File photoFile = null;
try {
	File storageDir = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES); // 保存到公共共享空间内,可以在资源管理器中找到,其他应用也可以直接访问
	// File storageDir = activity.getExternalFilesDir(Environment.DIRECTORY_PICTURES); 将图片放到应用专属空间内,其他的应用无法访问,也无法查看。随app的卸载而删除
	String imageFileName = "fileName";
	photoFile = File.createTempFile(imageFileName,  ".jpg", storageDir );
	// String filePath = photoFile.getAbsolutePath(); 获取到图片的路径,可以在onActivityResult中根据这个路径去获取图片
} catch (IOException e) {
    e.printStackTrace();
}
if (photoFile != null) {
    intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    intent.putExtra(MediaStore.EXTRA_OUTPUT,
            FileProvider.getUriForFile(activity,
                    activity.getApplicationContext().getPackageName() + ".provider", photoFile));
    activity.startActivityForResult(intent, 10010);
}

上面就是改完之后的代码,我自测了一下,在android11、android10、android7和android5的版本上运行都是没有问题的。
如果有朋友发现那个地方不对请及时和我联系,我会及时更改,避免对更多的人造成影响。谢谢!

Android11相机返回的Intent==null

Intent==null的原因就一个,就是在创建相机跳转Intent的时候你传入了文件的存储地址,所以在拍完照回调的时候就没有返回Intent。

intent.putExtra(MediaStore.EXTRA_OUTPUT,uri);// 设置照片的存储位置,如果删除此行,在onActivityResult中返回的Intent是有值的,如果此行存在则Intent==null

就是因为这行代码所以回调的Intent的才为null,如果删除这行代码,则在回调中可以可以通过intent获取到拍照的文件

if (requestCode == 10010 && resultCode == RESULT_OK) {
    Bundle extras = data.getExtras();
    Bitmap imageBitmap = (Bitmap) extras.get("data");
    imageView.setImageBitmap(imageBitmap);
}

这个地方获取的图片官方介绍说是个缩略图,如果要获取大图的是需要传入自己设置的存储路径,下面是官方的原话
获取缩略图
获取完整的图片
如果各位看官发现我博客中的纰漏或者错误的话,请及时和我联系,避免影响到更多人。
谢谢!
谢谢大家的观看

附:官方文档

  • 7
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
要创建一个Android Studio相机应用程序,可以参考以下步骤: 1. 首先,在Android Studio创建一个新的项目。在项目创建过程,选择空白活动模板。 2. 在项目的`AndroidManifest.xml`文件添加相机权限。可以使用以下代码片段添加相机权限: ``` <uses-permission android:name="android.permission.CAMERA" /> ``` 这将确保应用程序具有访问相机的权限。 3. 在应用程序的布局文件,添加一个`SurfaceView`用于显示相机预览。可以使用以下代码片段添加`SurfaceView`: ``` <SurfaceView android:id="@+id/surfaceView" android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 4. 在活动类,获取相机实例并在`SurfaceView`上显示相机预览。可以使用以下代码片段完成这一步骤: ``` // 获取相机实例 Camera camera = Camera.open(); // 设置相机预览显示在SurfaceView上 SurfaceView surfaceView = findViewById(R.id.surfaceView); SurfaceHolder surfaceHolder = surfaceView.getHolder(); surfaceHolder.addCallback(new SurfaceHolder.Callback() { @Override public void surfaceCreated(SurfaceHolder holder) { try { camera.setPreviewDisplay(holder); camera.startPreview(); } catch (IOException e) { e.printStackTrace(); } } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { // 预览大小发生变化时的处理逻辑 } @Override public void surfaceDestroyed(SurfaceHolder holder) { camera.stopPreview(); camera.release(); } }); ``` 这将使相机预览显示在`SurfaceView`上,并在活动生命周期内管理相机的打开和关闭。 5. 可以根据需求添加更多功能,例如拍照和录像等。可以使用Camera类的方法来实现这些功能。 这是一个简单的Android Studio相机应用程序的基本框架。根据需要可以进行更多的定制和功能添加。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [android studio 8.1.0 相机调用](https://blog.csdn.net/weixin_37997371/article/details/82897195)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [Android(Java) 的书店应用程序与源代码](https://download.csdn.net/download/qq_37270421/88268058)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值