热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android部分手机拍照后获取的图片被旋转问题的解决方法

这篇文章主要为大家详细介绍了Android部分手机拍照后获取的图片被旋转问题的解决方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

调用Android系统拍照功能后,三星手机拍摄后的照片被旋转了90度,横着拍给你变成竖的,竖的拍给你变成横的。其它品牌的手机都是正常的,就三星出现这个怪事。

在Android适配上,我原来一直以为国内的小米手机够奇葩了,结果还有更奇葩的!你说你没事旋转照片干啥,实在是猜不透其居心何在,纯粹是在给开发者制造麻烦啊!

解决办法是获取到拍照后照片被旋转的角度,再旋转回去就好了。

具体思路:
1、首先在调用拍照方法时,保存拍照后的相片原图,得到原图路径,(PhotoBitmapUtils是我自己写的一个工具类)

String fileName = ""; 
/** 
 * 启动相机拍照 
 */ 
private void addBitmapShoots() { 
 Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); 
 // 设置图片要保存的 根路径+文件名 
 fileName = PhotoBitmapUtils.getPhotoFileName(getContext()); 
 File file = new File(fileName); 
 if (!file.exists()) { 
  try { 
   file.createNewFile(); 
  } catch (IOException e) { 
   e.printStackTrace(); 
  } 
 } 
 intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file)); 
 startActivityForResult(intent, OPEN_CAMERA); 
} 

2、在获取相机返回的回调方法onActivityResult()中,修复被旋转的图片并取得修复后的图片路径,有了这个路径后就可以展示出来了

@Override 
public void onActivityResult(int requestCode, int resultCode, Intent data) { 
 super.onActivityResult(requestCode, resultCode, data); 
 // 获取相机拍照返回 
 if (resultCode == Activity.RESULT_OK && requestCode == OPEN_CAMERA) { 
  // 得到修复后的照片路径 
  String filepath = PhotoBitmapUtils.amendRotatePhoto(fileName, getContext()); 
 } 
} 

PhotoBitmapUtils类:

/** 
 * 集合一些图片工具 
 * 
 * Created by zhuwentao on 2016-07-22. 
 */ 
public class PhotoBitmapUtils { 
 
 /** 
  * 存放拍摄图片的文件夹 
  */ 
 private static final String FILES_NAME = "/MyPhoto"; 
 /** 
  * 获取的时间格式 
  */ 
 public static final String TIME_STYLE = "yyyyMMddHHmmss"; 
 /** 
  * 图片种类 
  */ 
 public static final String IMAGE_TYPE = ".png"; 
 
 // 防止实例化 
 private PhotoBitmapUtils() { 
 } 
 
 /** 
  * 获取手机可存储路径 
  * 
  * @param context 上下文 
  * @return 手机可存储路径 
  */ 
 private static String getPhoneRootPath(Context context) { 
  // 是否有SD卡 
  if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED) 
    || !Environment.isExternalStorageRemovable()) { 
   // 获取SD卡根目录 
   return context.getExternalCacheDir().getPath(); 
  } else { 
   // 获取apk包下的缓存路径 
   return context.getCacheDir().getPath(); 
  } 
 } 
 
 /** 
  * 使用当前系统时间作为上传图片的名称 
  * 
  * @return 存储的根路径+图片名称 
  */ 
 public static String getPhotoFileName(Context context) { 
  File file = new File(getPhoneRootPath(context) + FILES_NAME); 
  // 判断文件是否已经存在,不存在则创建 
  if (!file.exists()) { 
   file.mkdirs(); 
  } 
  // 设置图片文件名称 
  SimpleDateFormat format = new SimpleDateFormat(TIME_STYLE, Locale.getDefault()); 
  Date date = new Date(System.currentTimeMillis()); 
  String time = format.format(date); 
  String photOname= "/" + time + IMAGE_TYPE; 
  return file + photoName; 
 } 
 
 /** 
  * 保存Bitmap图片在SD卡中 
  * 如果没有SD卡则存在手机中 
  * 
  * @param mbitmap 需要保存的Bitmap图片 
  * @return 保存成功时返回图片的路径,失败时返回null 
  */ 
 public static String savePhotoToSD(Bitmap mbitmap, Context context) { 
  FileOutputStream outStream = null; 
  String fileName = getPhotoFileName(context); 
  try { 
   outStream = new FileOutputStream(fileName); 
   // 把数据写入文件,100表示不压缩 
   mbitmap.compress(Bitmap.CompressFormat.PNG, 100, outStream); 
   return fileName; 
  } catch (Exception e) { 
   e.printStackTrace(); 
   return null; 
  } finally { 
   try { 
    if (outStream != null) { 
     // 记得要关闭流! 
     outStream.close(); 
    } 
    if (mbitmap != null) { 
     mbitmap.recycle(); 
    } 
   } catch (Exception e) { 
    e.printStackTrace(); 
   } 
  } 
 } 
 
 /** 
  * 把原图按1/10的比例压缩 
  * 
  * @param path 原图的路径 
  * @return 压缩后的图片 
  */ 
 public static Bitmap getCompressPhoto(String path) { 
  BitmapFactory.Options optiOns= new BitmapFactory.Options(); 
  options.inJustDecodeBounds = false; 
  options.inSampleSize = 10; // 图片的大小设置为原来的十分之一 
  Bitmap bmp = BitmapFactory.decodeFile(path, options); 
  optiOns= null; 
  return bmp; 
 } 
 
 /** 
  * 处理旋转后的图片 
  * @param originpath 原图路径 
  * @param context 上下文 
  * @return 返回修复完毕后的图片路径 
  */ 
 public static String amendRotatePhoto(String originpath, Context context) { 
 
  // 取得图片旋转角度 
  int angle = readPictureDegree(originpath); 
 
  // 把原图压缩后得到Bitmap对象 
  Bitmap bmp = getCompressPhoto(originpath);; 
 
  // 修复图片被旋转的角度 
  Bitmap bitmap = rotaingImageView(angle, bmp); 
 
  // 保存修复后的图片并返回保存后的图片路径 
  return savePhotoToSD(bitmap, context); 
 } 
 
 /** 
  * 读取照片旋转角度 
  * 
  * @param path 照片路径 
  * @return 角度 
  */ 
 public static int readPictureDegree(String path) { 
  int degree = 0; 
  try { 
   ExifInterface exifInterface = new ExifInterface(path); 
   int orientation = exifInterface.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL); 
   switch (orientation) { 
    case ExifInterface.ORIENTATION_ROTATE_90: 
     degree = 90; 
     break; 
    case ExifInterface.ORIENTATION_ROTATE_180: 
     degree = 180; 
     break; 
    case ExifInterface.ORIENTATION_ROTATE_270: 
     degree = 270; 
     break; 
   } 
  } catch (IOException e) { 
   e.printStackTrace(); 
  } 
  return degree; 
 } 
 
 /** 
  * 旋转图片 
  * @param angle 被旋转角度 
  * @param bitmap 图片对象 
  * @return 旋转后的图片 
  */ 
 public static Bitmap rotaingImageView(int angle, Bitmap bitmap) { 
  Bitmap returnBm = null; 
  // 根据旋转角度,生成旋转矩阵 
  Matrix matrix = new Matrix(); 
  matrix.postRotate(angle); 
  try { 
   // 将原始图片按照旋转矩阵进行旋转,并得到新的图片 
   returnBm = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true); 
  } catch (OutOfMemoryError e) { 
  } 
  if (returnBm == null) { 
   returnBm = bitmap; 
  } 
  if (bitmap != returnBm) { 
   bitmap.recycle(); 
  } 
  return returnBm; 
 } 
} 

在调用修复图片角度方法的时候需要注意,现在的手机像素越来越大,拍完后一张照片有近10M,所以我们需要对图片进行压缩处理。不然在保存图片时会等待挺久的,屏幕会黑一会。
参考文档1
参考文档2

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 本文详细介绍如何使用arm-eabi-gdb调试Android平台上的C/C++程序。通过具体步骤和实用技巧,帮助开发者更高效地进行调试工作。 ... [详细]
  • 深入理解 Oracle 存储函数:计算员工年收入
    本文介绍如何使用 Oracle 存储函数查询特定员工的年收入。我们将详细解释存储函数的创建过程,并提供完整的代码示例。 ... [详细]
  • 本文介绍如何在 Xcode 中使用快捷键和菜单命令对多行代码进行缩进,包括右缩进和左缩进的具体操作方法。 ... [详细]
  • 本文介绍了一款用于自动化部署 Linux 服务的 Bash 脚本。该脚本不仅涵盖了基本的文件复制和目录创建,还处理了系统服务的配置和启动,确保在多种 Linux 发行版上都能顺利运行。 ... [详细]
  • 在Linux系统中配置并启动ActiveMQ
    本文详细介绍了如何在Linux环境中安装和配置ActiveMQ,包括端口开放及防火墙设置。通过本文,您可以掌握完整的ActiveMQ部署流程,确保其在网络环境中正常运行。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • 如何在WPS Office for Mac中调整Word文档的文字排列方向
    本文将详细介绍如何使用最新版WPS Office for Mac调整Word文档中的文字排列方向。通过这些步骤,用户可以轻松更改文本的水平或垂直排列方式,以满足不同的排版需求。 ... [详细]
  • 本文总结了在使用Ionic 5进行Android平台APK打包时遇到的问题,特别是针对QRScanner插件的改造。通过详细分析和提供具体的解决方法,帮助开发者顺利打包并优化应用性能。 ... [详细]
  • 理解存储器的层次结构有助于程序员优化程序性能,通过合理安排数据在不同层级的存储位置,提升CPU的数据访问速度。本文详细探讨了静态随机访问存储器(SRAM)和动态随机访问存储器(DRAM)的工作原理及其应用场景,并介绍了存储器模块中的数据存取过程及局部性原理。 ... [详细]
  • 360SRC安全应急响应:从漏洞提交到修复的全过程
    本文详细介绍了360SRC平台处理一起关键安全事件的过程,涵盖从漏洞提交、验证、排查到最终修复的各个环节。通过这一案例,展示了360在安全应急响应方面的专业能力和严谨态度。 ... [详细]
  • 几何画板展示电场线与等势面的交互关系
    几何画板是一款功能强大的物理教学软件,具备丰富的绘图和度量工具。它不仅能够模拟物理实验过程,还能通过定量分析揭示物理现象背后的规律,尤其适用于难以在实际实验中展示的内容。本文将介绍如何使用几何画板演示电场线与等势面之间的关系。 ... [详细]
  • 本文介绍如何通过Windows批处理脚本定期检查并重启Java应用程序,确保其持续稳定运行。脚本每30分钟检查一次,并在需要时重启Java程序。同时,它会将任务结果发送到Redis。 ... [详细]
  • MySQL中枚举类型的所有可能值获取方法
    本文介绍了一种在MySQL数据库中查询枚举(ENUM)类型字段所有可能取值的方法,帮助开发者更好地理解和利用这一数据类型。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • 本文详细探讨了在Android 8.0设备上使用ChinaCock的TCCBarcodeScanner进行扫码时出现的应用闪退问题,并提供了解决方案。通过调整配置文件,可以有效避免这一问题。 ... [详细]
author-avatar
小呀小-儿郎
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有