Android开发之使用ExifInterface获取拍照后的图片属性

本文实例讲述了Android开发之使用ExifInterface获取拍照后的图片属性。分享给大家供大家参考,具体如下:

ExifInterface exif = new ExifInterface(file.getPath()); String widthStr = exif.getAttribute(ExifInterface.TAG_IMAGE_WIDTH); String heightStr = exif.getAttribute(ExifInterface.TAG_IMAGE_LENGTH); ...... /* * Copyright (C) 2007 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.media; import java.io.IOException; import java.text.ParsePosition; import java.text.SimpleDateFormat; import java.util.Date; import java.util.HashMap; import java.util.Map; import java.util.TimeZone; /** * This is a class for reading and writing Exif tags in a JPEG file. */ public class ExifInterface { // The Exif tag names /** Type is int. */ public static final String TAG_ORIENTATION = "Orientation"; /** Type is String. */ public static final String TAG_DATETIME = "DateTime"; /** Type is String. */ public static final String TAG_MAKE = "Make"; /** Type is String. */ public static final String TAG_MODEL = "Model"; /** Type is int. */ public static final String TAG_FLASH = "Flash"; /** Type is int. */ public static final String TAG_IMAGE_WIDTH = "ImageWidth"; /** Type is int. */ public static final String TAG_IMAGE_LENGTH = "ImageLength"; /** String. Format is "num1/denom1,num2/denom2,num3/denom3". */ public static final String TAG_GPS_LATITUDE = "GPSLatitude"; /** String. Format is "num1/denom1,num2/denom2,num3/denom3". */ public static final String TAG_GPS_LONGITUDE = "GPSLongitude"; /** Type is String. */ public static final String TAG_GPS_LATITUDE_REF = "GPSLatitudeRef"; /** Type is String. */ public static final String TAG_GPS_LONGITUDE_REF = "GPSLongitudeRef"; /** Type is String. */ public static final String TAG_GPS_TIMESTAMP = "GPSTimeStamp"; /** Type is String. */ public static final String TAG_GPS_DATESTAMP = "GPSDateStamp"; /** Type is int. */ public static final String TAG_WHITE_BALANCE = "WhiteBalance"; /** Type is rational. */ public static final String TAG_FOCAL_LENGTH = "FocalLength"; /** Type is String. Name of GPS processing method used for location finding. */ public static final String TAG_GPS_PROCESSING_METHOD = "GPSProcessingMethod"; // Constants used for the Orientation Exif tag. public static final int ORIENTATION_UNDEFINED = 0; public static final int ORIENTATION_NORMAL = 1; public static final int ORIENTATION_FLIP_HORIZONTAL = 2; // left right reversed mirror public static final int ORIENTATION_ROTATE_180 = 3; public static final int ORIENTATION_FLIP_VERTICAL = 4; // upside down mirror public static final int ORIENTATION_TRANSPOSE = 5; // flipped about top-left <--> bottom-right axis public static final int ORIENTATION_ROTATE_90 = 6; // rotate 90 cw to right it public static final int ORIENTATION_TRANSVERSE = 7; // flipped about top-right <--> bottom-left axis public static final int ORIENTATION_ROTATE_270 = 8; // rotate 270 to right it // Constants used for white balance public static final int WHITEBALANCE_AUTO = 0; public static final int WHITEBALANCE_MANUAL = 1; private static SimpleDateFormat sFormatter; static { System.loadLibrary("exif"); sFormatter = new SimpleDateFormat("yyyy:MM:dd HH:mm:ss"); sFormatter.setTimeZone(TimeZone.getTimeZone("UTC")); } private String mFilename; private HashMap<String, String> mAttributes; private boolean mHasThumbnail; // Because the underlying implementation (jhead) uses static variables, // there can only be one user at a time for the native functions (and // they cannot keep state in the native code across function calls). We // use sLock to serialize the accesses. private static Object sLock = new Object(); /** * Reads Exif tags from the specified JPEG file. */ public ExifInterface(String filename) throws IOException { mFilename = filename; loadAttributes(); } /** * Returns the value of the specified tag or {@code null} if there * is no such tag in the JPEG file. * * @param tag the name of the tag. */ public String getAttribute(String tag) { return mAttributes.get(tag); } /** * Returns the integer value of the specified tag. If there is no such tag * in the JPEG file or the value cannot be parsed as integer, return * <var>defaultValue</var>. * * @param tag the name of the tag. * @param defaultValue the value to return if the tag is not available. */ public int getAttributeInt(String tag, int defaultValue) { String value = mAttributes.get(tag); if (value == null) return defaultValue; try { return Integer.valueOf(value); } catch (NumberFormatException ex) { return defaultValue; } } /** * Returns the double value of the specified rational tag. If there is no * such tag in the JPEG file or the value cannot be parsed as double, return * <var>defaultValue</var>. * * @param tag the name of the tag. * @param defaultValue the value to return if the tag is not available. */ public double getAttributeDouble(String tag, double defaultValue) { String value = mAttributes.get(tag); if (value == null) return defaultValue; try { int index = value.indexOf("/"); if (index == -1) return defaultValue; double denom = Double.parseDouble(value.substring(index + 1)); if (denom == 0) return defaultValue; double num = Double.parseDouble(value.substring(0, index)); return num / denom; } catch (NumberFormatException ex) { return defaultValue; } } /** * Set the value of the specified tag. * * @param tag the name of the tag. * @param value the value of the tag. */ public void setAttribute(String tag, String value) { mAttributes.put(tag, value); } /** * Initialize mAttributes with the attributes from the file mFilename. * * mAttributes is a HashMap which stores the Exif attributes of the file. * The key is the standard tag name and the value is the tag's value: e.g. * Model -> Nikon. Numeric values are stored as strings. * * This function also initialize mHasThumbnail to indicate whether the * file has a thumbnail inside. */ private void loadAttributes() throws IOException { // format of string passed from native C code: // "attrCnt attr1=valueLen value1attr2=value2Len value2..." // example: // "4 attrPtr ImageLength=4 1024Model=6 FooImageWidth=4 1280Make=3 FOO" mAttributes = new HashMap<String, String>(); String attrStr; synchronized (sLock) { attrStr = getAttributesNative(mFilename); } // get count int ptr = attrStr.indexOf(' '); int count = Integer.parseInt(attrStr.substring(0, ptr)); // skip past the space between item count and the rest of the attributes ++ptr; for (int i = 0; i < count; i++) { // extract the attribute name int equalPos = attrStr.indexOf('=', ptr); String attrName = attrStr.substring(ptr, equalPos); ptr = equalPos + 1; // skip past = // extract the attribute value length int lenPos = attrStr.indexOf(' ', ptr); int attrLen = Integer.parseInt(attrStr.substring(ptr, lenPos)); ptr = lenPos + 1; // skip pas the space // extract the attribute value String attrValue = attrStr.substring(ptr, ptr + attrLen); ptr += attrLen; if (attrName.equals("hasThumbnail")) { mHasThumbnail = attrValue.equalsIgnoreCase("true"); } else { mAttributes.put(attrName, attrValue); } } } /** * Save the tag data into the JPEG file. This is expensive because it involves * copying all the JPG data from one file to another and deleting the old file * and renaming the other. It's best to use {@link #setAttribute(String,String)} * to set all attributes to write and make a single call rather than multiple * calls for each attribute. */ public void saveAttributes() throws IOException { // format of string passed to native C code: // "attrCnt attr1=valueLen value1attr2=value2Len value2..." // example: // "4 attrPtr ImageLength=4 1024Model=6 FooImageWidth=4 1280Make=3 FOO" StringBuilder sb = new StringBuilder(); int size = mAttributes.size(); if (mAttributes.containsKey("hasThumbnail")) { --size; } sb.append(size + " "); for (Map.Entry<String, String> iter : mAttributes.entrySet()) { String key = iter.getKey(); if (key.equals("hasThumbnail")) { // this is a fake attribute not saved as an exif tag continue; } String val = iter.getValue(); sb.append(key + "="); sb.append(val.length() + " "); sb.append(val); } String s = sb.toString(); synchronized (sLock) { saveAttributesNative(mFilename, s); commitChangesNative(mFilename); } } /** * Returns true if the JPEG file has a thumbnail. */ public boolean hasThumbnail() { return mHasThumbnail; } /** * Returns the thumbnail inside the JPEG file, or {@code null} if there is no thumbnail. * The returned data is in JPEG format and can be decoded using * {@link android.graphics.BitmapFactory#decodeByteArray(byte[],int,int)} */ public byte[] getThumbnail() { synchronized (sLock) { return getThumbnailNative(mFilename); } } /** * Stores the latitude and longitude value in a float array. The first element is * the latitude, and the second element is the longitude. Returns false if the * Exif tags are not available. */ public boolean getLatLong(float output[]) { String latValue = mAttributes.get(ExifInterface.TAG_GPS_LATITUDE); String latRef = mAttributes.get(ExifInterface.TAG_GPS_LATITUDE_REF); String lngValue = mAttributes.get(ExifInterface.TAG_GPS_LONGITUDE); String lngRef = mAttributes.get(ExifInterface.TAG_GPS_LONGITUDE_REF); if (latValue != null && latRef != null && lngValue != null && lngRef != null) { output[0] = convertRationalLatLonToFloat(latValue, latRef); output[1] = convertRationalLatLonToFloat(lngValue, lngRef); return true; } else { return false; } } /** * Returns number of milliseconds since Jan. 1, 1970, midnight. * Returns -1 if the date time information if not available. * @hide */ public long getDateTime() { String dateTimeString = mAttributes.get(TAG_DATETIME); if (dateTimeString == null) return -1; ParsePosition pos = new ParsePosition(0); try { Date datetime = sFormatter.parse(dateTimeString, pos); if (datetime == null) return -1; return datetime.getTime(); } catch (IllegalArgumentException ex) { return -1; } } /** * Returns number of milliseconds since Jan. 1, 1970, midnight UTC. * Returns -1 if the date time information if not available. * @hide */ public long getGpsDateTime() { String date = mAttributes.get(TAG_GPS_DATESTAMP); String time = mAttributes.get(TAG_GPS_TIMESTAMP); if (date == null || time == null) return -1; String dateTimeString = date + ' ' + time; if (dateTimeString == null) return -1; ParsePosition pos = new ParsePosition(0); try { Date datetime = sFormatter.parse(dateTimeString, pos); if (datetime == null) return -1; return datetime.getTime(); } catch (IllegalArgumentException ex) { return -1; } } private static float convertRationalLatLonToFloat( String rationalString, String ref) { try { String [] parts = rationalString.split(","); String [] pair; pair = parts[0].split("/"); int degrees = (int) (Float.parseFloat(pair[0].trim()) / Float.parseFloat(pair[1].trim())); pair = parts[1].split("/"); int minutes = (int) ((Float.parseFloat(pair[0].trim()) / Float.parseFloat(pair[1].trim()))); pair = parts[2].split("/"); float seconds = Float.parseFloat(pair[0].trim()) / Float.parseFloat(pair[1].trim()); float result = degrees + (minutes / 60F) + (seconds / (60F * 60F)); if ((ref.equals("S") || ref.equals("W"))) { return -result; } return result; } catch (RuntimeException ex) { // if for whatever reason we can't parse the lat long then return // null return 0f; } } private native boolean appendThumbnailNative(String fileName, String thumbnailFileName); private native void saveAttributesNative(String fileName, String compressedAttributes); private native String getAttributesNative(String fileName); private native void commitChangesNative(String fileName); private native byte[] getThumbnailNative(String fileName); }

更多关于Android开发相关内容感兴趣的读者可查看本站专题:《Android开发入门与进阶教程》及《Android图形与图像处理技巧总结》

希望本文所述对大家Android程序设计有所帮助。

时间: 2024-10-27 22:10:16

Android开发之使用ExifInterface获取拍照后的图片属性的相关文章

Android开发之使用ExifInterface获取拍照后的图片属性_Android

本文实例讲述了Android开发之使用ExifInterface获取拍照后的图片属性.分享给大家供大家参考,具体如下: ExifInterface exif = new ExifInterface(file.getPath()); String widthStr = exif.getAttribute(ExifInterface.TAG_IMAGE_WIDTH); String heightStr = exif.getAttribute(ExifInterface.TAG_IMAGE_LENGT

Android实现从本地图库/相机拍照后裁剪图片并设置头像_Android

先给大家展示效果图: 代码部分: 布局代码(其实就是两个按钮和一个ImageView来显示头像) <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="v

Android实现从本地图库/相机拍照后裁剪图片并设置头像

玩qq或者是微信的盆友都知道,这些聊天工具里都要设置头像,一般情况下大家的解决办法是从本地图库选择图片或是从相机拍照,然后根据自己的喜爱截取图片.上述过程已经实现好了,最后一步我加上了把截取好的图片在保存到本地的操作,来保存头像.为了大家需要,下面脚本之家小编把完整的代码贴出来供大家参考. 先给大家展示效果图: 代码部分: 布局代码(其实就是两个按钮和一个ImageView来显示头像) <LinearLayout xmlns:android="http://schemas.android.

Android 开发手机(三星)拍照应用照片旋转问题解决办法

Android 开发手机(三星)拍照应用照片旋转问题解决办法 最近解决了一个令我头疼好久的问题,就是三星手机拍照图片旋转的问题,项目中有上传图片的功能,那么涉及到拍照,从相册中选择图片,别的手机都ok没有问题,唯独三星的手机拍照之后,你会很清楚的看到会把照片旋转一下,然后你根据路径找到的图片就是已经被旋转的了,解决办法终于被我找到了.我们可以根据图片的路径读取照片exif(Exchangeable Image File 可交换图像文件)信息中的旋转角度 根据调试,可以清楚的发现三星手机拍照的图片

Android开发中MotionEvent坐标获取方法分析_Android

本文实例讲述了Android开发中MotionEvent坐标获取方法.分享给大家供大家参考,具体如下: Android MotionEvent中getX()与getRawX()都是获取屏幕坐标(横),但二者又有区别getX()           :   是获取相对当前控件(View)的坐标 getRawX()   :   是获取相对显示屏幕左上角的坐标 演示示例代码 Java代码: public class MainActivity extends Activity implements On

android 开发 怎么向服务器上传视频和图片文件啊!

问题描述 android 开发 怎么向服务器上传视频和图片文件啊! 做到这里卡住了 希望会的大神指导下! 解决方案 http://download.csdn.net/download/tianzhu15/4568311 解决方案二: http://blog.csdn.net/jdsjlzx/article/details/44160603 解决方案三: 上传图片/文件到服务器java 上传文件以及各种视频图片!

Android开发中LinearLayout布局技巧及layout中drawable属性区别

先介绍drawable属性的区别,这个算是比较简单的,但是还是有一点点的小细节需要进行说明,drawable有五个文件夹,分别为hdpi,ldpi,mdpi,xdpi,xxdpi,这五个文件夹想必大家都知道,其实就是为了适应不同分辨率,由于手机分辨率的不同,因此我们的图片需要适应不同手机的分辨率...hdpi:480x800   mdpi:480x320   ldpi:320x240xdpi:1280x720 xxdpi 1920x1280其实这个数字并不是非常精确的,只是说明每一个阶段都有一个

Android开发之一种简单的异步加载图片方法

首先说明的是,该方法已经被我抛弃了.之前用它,发现加载速度不好.具体没怎么细心的看. 现在我用volley了.拿出来只是给大家批判的. package com.souya.seller.util.ex; import java.io.File; import java.io.FileInputStream; import java.io.InputStream; import java.lang.ref.SoftReference; import java.util.HashMap; impor

android开发-想用socket从服务端发送图片到客户端,但一直出错,求原因

问题描述 想用socket从服务端发送图片到客户端,但一直出错,求原因 开发-想用socket从服务端发送图片到客户端,但一直出错,求原因-socket客户端和服务端"> 如图所示,本人新人,求大神解答 解决方案 空指针从无,TestActivity的64行 解决方案二: 空指针 socket 不能执行关闭操作(其实是不能操作任何操作) 最好在操作前增加判断:if(null != socket) {} 在 {} 中做处理. 解决方案三: 应该是创建socket失败了. socket不能放在