php正确处理照片的旋转问题

前言

iPhone和一些数码相机在拍照的时候往往会在图片里面加入很多的照片信息(exif),比如拍照时间、光圈大小、曝光时间、GSP地理信息以及拍摄时相机倾斜状态等等,这些信息往往会提供给某些系统来对图片进行正确的显示,但是我们在对处理图片的时候通常会丢失掉那些图片信息,这就会导致显示出现问题,很典型的一个显示错误就是对Iphone拍摄的照片进行剪裁之后发现图片旋转了90度,这个时候就需要对图片进行相应的处理。

背景

博主是在一个小项目中遇到的这个问题,这个项目虽然小,但是涉及到的各种坑真的可以写一篇比较长的技术文章了,这里先不描述,后面单独发一篇关于这个项目的文章,把所涉及到的知识点以及坑点一一描述。

这个小项目中遇到的问题之一就是,我需要使用html5的canvas把读取到的图片进行剪裁,然后渲染到画布上。很简单的一个需求,但是在Iphone上测试经常会看到剪裁完之后图片自动旋转了90度,安卓机上不存在这个问题。

分析

之前就遇到过这个问题,iPhone拍摄的照片复制到win7上打开图片就是一个倒过来的,而在手机上看却没有问题,这就是因为exif信息在搞鬼,简单点说exif就是保留了图片的拍摄参数,显示的时候程序会读取图片的exif信息,如果exif信息不存在,那么图片就是按照最原始的形式显示,如果有exif信息,程序就会按照exif来。

这里博主遇到的情况是图片在进行剪裁的时候导致exif信息丢失,本来程序读取exif中图片要求顺时针旋转90°,现在exif丢失,程序读不到,默认图片不用旋转,那么我们看到的图片就是倒着的,这里应该怎么解决呢?

在这个项目中,由于读取的图片存在跨域问题,不能直接对本地的图片进行裁剪,所以是先上传到远程服务器,生成一个远程的地址,然后再用canvas读取远程地址,这样就解决了跨域问题。

现在摆在博主面前就有两种解决方案了

只用exif.js本地读取图片的exif信息,直接在客户端进行判断

上传到服务器的时候对图片进行处理,返回的远程地址已经是处理过的图片

对这两种方案进行对比,第一种方案看起来确实挺适合,直接在前端进行处理,这里会非常节省服务器资源,但是缺点也很明显,canvas对图片的渲染有一定的限制,当图片过大的时候渲染就会失败,而在这个项目中客户要求最后生成的图片一定要高清大图,上传的照片一般都有3~5M大小,这么大的图片大多数时候canvas都无法处理。

exif.js的原理其实比较简单,就是把图片转换成base64字符串,对其中的信息进行解析,所以一旦要对大图进行处理的时候就会出现问题,数据量太大,导致失败。

第一种方案不可行就只能选择第二种方案了,项目后端使用php做中转处理,前面说了需要生成一个远程地址,php负责把接收到的图片存储在远程服务器,然后返回一个地址给前端,这里只需要在存储之前对图片做处理就可以解决这个问题。

解决

首先给出完整代码,然后对代码进行解释

include_once ("../weixin/jssdk.php");
$jssdk = new JSSDK("wx**************", "******************************");
$access_token = $jssdk->getAccessToken();

$media_id = $_POST['i'];
$savePathFile = '/upload/temps/'.date('YmdHis').rand(1000,9999).'.jpg';
$targetName = __DIR__.$savePathFile;

$str = "http://file.api.weixin.qq.com/cgi-bin/media/get?access_token=".$access_token."&media_id=".$media_id;
$image = imagecreatefromstring(file_get_contents($str));
$exif = exif_read_data($str);

if(!empty($exif['Orientation'])) {
    switch($exif['Orientation']) {
        case 8:
            $image = imagerotate($image,90,0);
            break;
        case 3:
            $image = imagerotate($image,180,0);
            break;
        case 6:
            $image = imagerotate($image,-90,0);
            break;
    }
}

imagejpeg($image,$targetName);
echo json_encode(array("code"=>0,'d'=>$exif['Orientation'],"path"=>$savePathFile));
看起来确实比较简单,主要还是调用了一些内部函数,处理起来就方便多了。

这个项目使用了微信的上传接口,所以php需要从微信的临时服务器上把用户上传的图片取回来,通过file_get_contents方法就能快速拿到,当然也可以用curl来做。接着使用imagecreatefromstring创建一个图片的缓存,正常情况下如果不需要对图片进行处理,那么接下来就可以直接存储了,这里我们还需要对图片进行一些操作。

通过 exif_read_data 方法可以直接读取到图片的exif信息。

注意:exif_read_data 这个方法是exif扩展里面的方法,如果不能执行,请检查扩展是否安装,是否开启等。

exif模块里面还有很多有意思的方法,可以拿来做很多有用的东西,比如拿来批量采集并分析图片,提取图片信息等,这里推荐一个好玩的网站 https://exif.cn/

if(!empty($exif['Orientation'])) {
    switch($exif['Orientation']) {
        case 8:
            $image = imagerotate($image,90,0);
            break;
        case 3:
            $image = imagerotate($image,180,0);
            break;
        case 6:
            $image = imagerotate($image,-90,0);
            break;
    }
}
这一段就很容易看懂了,就是判断图片的旋转状态,对图片进行旋转处理imagerotate方法很好用,当然同类的还有好多函数,可以去了解下,如果你正在做图片处理,这些函数应该会有很大的帮助。

最后使用 imagejpeg 方法把处理过的图片写入到磁盘,然后返回一个包含远程地址的json到前端。

注意:我这里没有做异常捕获,正常情况下文件io操作都必须要做异常捕获,这里的代码只为了说明使用方法,所以省略了这一步。

后记

在这个小项目中,最为常见的问题就是在php函数的使用上,同一种效果可以使用多个函数进行处理,选择一个简洁高效的函数非常重要,在使用第三方扩展的时候务必确认扩展是否安装,扩展依赖的插件是否安装,是否已经开启,是否有其他额外的附加条件等。

前端方面需要知道所使用的js插件有那些附加功能,如果api文章不够清晰可以直接打开插件源码,通常情况下,一个出色的插件往往会在未压缩的代码里面写明所有接口的使用方法,以及注意事项,使用条件等。

此外,还需要对不常见的情况进行判定,当出现一些诡异bug的时候就应该考虑是否是因为当前所给的参数不符合规范,或者超出了规范允许的范围,上面说道的图片转base64格式的大小限制就是一个很难被发现的问题,博主在调试的时候对生成的数据进行观察才发现,当出现大图的时候就会转换失败。

时间: 2024-10-27 02:26:24

php正确处理照片的旋转问题的相关文章

android-三星 note II 调用系统拍照获取到的照片是旋转的

问题描述 三星 note II 调用系统拍照获取到的照片是旋转的 拍完照好像旋转了,但到文件管理器中看到的图片并不是旋转的,这是怎么回事,难道还要手动旋转下? 解决方案 三星note2 android调用系统拍照空指针异常三星note2 android调用系统拍照空指针异常关于三星手机拍照 获取的照片旋转处理 解决方案二: 设置旋转为不旋转就可以了写死

用iSee软件轻松管理假日海量照片

假日外出游玩归来,拍摄了海量的数码照片.该如何处理?专家向你支招: 1)从相机储存卡导入照片 2)筛除"败片" 3)裁剪照片,获得最佳构图 4)上传分享 1)从相机储存卡导入照片 数码相机一般都采用SD或CF储存卡,通过读卡器连接电脑的USB端口,就可以读取到照片.对于电脑文件拷贝还不太熟悉的朋友来说,iSee的照片导入功能可以帮上不小的忙. 点击菜单[文件]->[获取/导出],选择[从移动设备获取图象],弹出以下界面: 按上图所示设置"存放位置"为"

PS CS5+LR对曝光过度照片的后期修复教程

原片: 修复后: 最终效果: 注意:本教程所使用的原片是RAW文件,如果是JPG文件则不会有这么大的调整空间,只有RAW文件才能更好的做后期调整,所以建议平时在拍重要照片的时候 尽量RAW格式存储. 一.LR前处理   1.在Lightroom 3.2(简称:LR)中导入原片,进入"修改照片"页面,首先在执行"相机校准"选择"Camera Landscape"(相机风景模式). 选择这种色彩风格也是为了增强自然色彩. 2.镜头校正: LR3.2集

美图看看和Windows 7照片查看器哪个好?

  很多人咋浏览照片时,都是使用的Windows 7自带的图片浏览器,因为这样方便,且在不占内存的情况下,能最大满足用户的图片浏览需求.不过随着图片格式的增多,买很多用户为了能浏览更多格式的图片,也会安装第三方的看图工具.今天就为大家带来一款名为"美图看看"的看图软件在Windows 7系统下,和系统自带的照片查看器之间的应用对比,希望对正在使用Windows 7系统的朋友有个使用参考. 一.调用简单的Windows 7照片查看器 如果只是暂时的查看电脑中存放的图片,那Windows

PhotoShop打造高端宫廷婚纱照片HDR效果调色教程

不过我用的方法比较另类,完全是自已的一套思路,大家可以自行判断对错. 这次我说的是用一张片子,通过后期调整来模拟HDR效果,不是通过摄影技术来制作HDR照片. 首先要知道不是所有片子都适合调HDR效果的,首先片子当然得是一张拍摄正确的照片(曝光.色温.构图都是OK的),然后片子的内容一定要很丰富, 画面内的元素和色彩太单一就能难好看,因为HDR效果就是把画面中更多细节呈现出来,没什么元素色彩细节可看就没意义了. 所以实际上案例的这张片子就不适合做很夸张的HDR效果,片子颜色比较单一,最后挑调完会

解决android有的手机拍照后上传图片被旋转的问题_Android

需求:做仿新浪发微博的项目,能够上传图片还有两外一个项目用到手机拍摄图片,这两个都需要把图片上传到服务器 遇到问题:有的手机拍摄的图片旋转90度,有的图片旋转了180度,有的手机是正常的,服务器要求的是正的,这样问题就来了,不能用户发个照片在微博上看到的是被旋转了的啊,另外一个项目里旋转了的图片直接匹配出现问题,这个更严重. 解决:开始的时候在网上没有找到很好的解决办法,谷歌百度的搜了一通,想到第一种解决方式,当手机拍照结束,在返回结果处理里面立即跳转到一个新的页面,在新的页面里让用户自己手动去

iOS拍的图片上传到服务器(阿里云)旋转

用ghost发文章(发到阿里云的CDN)的时候,发现上传的图片旋转了.比如这张图: 但是,在我的Mac本地上是正常的.像这样子的: 打开预览的效果也是正常的. 然后,我傻傻以为是Ghost的问题.其实并不是. 这里,要开始引出EXIF这个概念. EXIF EXIF(Exchangeable Image File)是"可交换图像文件"的缩写,当中包含了专门为数码相机的照片而定制的元数据,可以记录数码照片的拍摄参数.缩略图及其他属性信息. 图像一旦被修改,Exif 信息可能会永久丢失,故编

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

调用Android系统拍照功能后,三星手机拍摄后的照片被旋转了90度,横着拍给你变成竖的,竖的拍给你变成横的.其它品牌的手机都是正常的,就三星出现这个怪事. 在Android适配上,我原来一直以为国内的小米手机够奇葩了,结果还有更奇葩的!你说你没事旋转照片干啥,实在是猜不透其居心何在,纯粹是在给开发者制造麻烦啊! 解决办法是获取到拍照后照片被旋转的角度,再旋转回去就好了. 具体思路: 1.首先在调用拍照方法时,保存拍照后的相片原图,得到原图路径,(PhotoBitmapUtils是我自己写的一个

MSDN 杂志:UI 前沿技术 - WPF 中的多点触控操作事件

原文  MSDN 杂志:UI 前沿技术 - WPF 中的多点触控操作事件 UI 前沿技术 WPF 中的多点触控操作事件 Charles Petzold 下载代码示例 就在过去几年,多点触控还只是科幻电影中表现未来主义的一种重要手法,现在俨然已经成为主流的用户界面技术. 多点触控显示屏现在成了新型智能手机和 Tablet 计算机的标准显示屏. 此外,它还可能在公共场所的计算机上普及,例如 Microsoft Surface 率先开发的网亭或桌面计算机. 实际存在的唯一不确定因素是多点触控在常规台式