
如果转换结果返回为nul,则可以认为后缀名不是有效的文件类型。
此方法实际是由查找对应的ContentProvider来获取文件类型,会先在本进程内查找,失败后通过ActivityManagerService跨进程查找。
首先要设置数据源,常用的数据设置有两种方式:
无论哪种设置方式,其本质都是获取文件描述符FileDescriptor,然后调用以下方法:
设置之后直接调用extractMetadata方法,传入keyCode为MediaMetadataRetriever.METADATA_KEY_MIMETYPE,即可得到文件的mime类型。
这个方法是JAVA 1.6版本以后提供的,但是在Android中需要Api 26以上才可以使用,方法中的参数path可以使用File内的toPath()方法获取。
5.1. 使用java中URLConnection提供的“类型猜测”方法查询:
此方法与方法一查询mime方法本质相同,最终都会调用MimeUtils.guessMimeTypeFromExtension方法。MimeUtils中提前缓存了后缀与mime类型的映射关系(先尝试从特定的用户表中加载,如果失败则加载默认的内置表),于是根据传入的后缀查询对应mime。
先读取数据流的前16个字节,根据这16个字节可以判断出绝大部分的文件类型。不过URLConnection下的这个方法只提供网络传输中常用的几种类型判断。
5.2. 根据前16个字节进行判断
以下只列出部分常用类型的应对关系,更多类型可以搜索“根据文件头获取文件类型”查询:
如果确定可转为Bitmap,可以直接使用BitmapFactory.Options中的outMimeType属性。
github: https://github.com/FirstLetterZ/Dependence/tree/master/file
依赖版本: 'io.github.firstletterz:tool-file:0.0.2'
2021-08-10
获取文件类型,一般的是列出目前所有的文件类型,根据表头进行相应判断,示例如下:
/*** 件头是位于文件开头的一段承担一定任务的数据,一般都在开头的部分。
* 头文件作为一种包含功能函数、数据接口声明的载体文件,用于保存程序的声明(declaration),而定义文件用于保存程序的实现 (implementation)。
* 为了解决在用户上传文件的时候在服务器端判断文件类型的问题,故用获取文件头的方式,直接读取文件的前几个字节,来判断上传文件是否符合格式。具体代码如下:
* Java代码 :
*
*/
package com.yonyou.sud.file
import java.io.FileInputStream
import java.io.IOException
import java.util.HashMap
/**
* 获取和判断文件头信息
*
* @author Sud
*
*/
public class GetTypeByHead {
//缓存文件头信息-文件头信息
public static final HashMap<String, String> mFileTypes = new HashMap<String, String>()
static {
// images
mFileTypes.put("FFD8FF", "jpg")
mFileTypes.put("89504E47", "png")
mFileTypes.put("47494638", "gif")
mFileTypes.put("49492A00", "tif")
mFileTypes.put("424D", "bmp")
//
mFileTypes.put("41433130", "dwg") // CAD
mFileTypes.put("38425053", "psd")
mFileTypes.put("7B5C727466", "rtf") // 日记本
mFileTypes.put("3C3F786D6C", "xml")
mFileTypes.put("68746D6C3E", "html")
mFileTypes.put("44656C69766572792D646174653A", "eml") // 邮件
mFileTypes.put("D0CF11E0", "doc")
mFileTypes.put("5374616E64617264204A", "mdb")
mFileTypes.put("252150532D41646F6265", "ps")
mFileTypes.put("255044462D312E", "pdf")
mFileTypes.put("504B0304", "docx")
mFileTypes.put("52617221", "rar")
mFileTypes.put("57415645", "wav")
mFileTypes.put("41564920", "avi")
mFileTypes.put("2E524D46", "rm")
mFileTypes.put("000001BA", "mpg")
mFileTypes.put("000001B3", "mpg")
mFileTypes.put("6D6F6F76", "mov")
mFileTypes.put("3026B2758E66CF11", "asf")
mFileTypes.put("4D546864", "mid")
mFileTypes.put("1F8B08", "gz")
}
/**
* 根据文件路径获取文件头信息
*
* @param filePath
* 文件路径
* @return 文件头信息
*/
public static String getFileType(String filePath){
System.out.println(getFileHeader(filePath))
System.out.println(mFileTypes.get(getFileHeader(filePath)))
return mFileTypes.get(getFileHeader(filePath))
}
/**
* 根据文件路径获取文件头信息
*
* @param filePath
* 文件路径
* @return 文件头信息
*/
public static String getFileHeader(String filePath){
FileInputStream is = null
String value = null
try {
is = new FileInputStream(filePath)
byte[] b = new byte[4]
/*int read() 从此输入流中读取一个数据字节。
*int read(byte[] b) 从此输入流中将最多 b.length 个字节的数据读入一个 byte 数组中。
* int read(byte[] b, int off, int len) 从此输入流中将最多 len 个字节的数据读入一个 byte 数组中。
*/
is.read(b, 0, b.length)
value = bytesToHexString(b)
} catch (Exception e){
} finally {
if (null != is){
try {
is.close()
} catch (IOException e){
}
}
}
return value
}
/**
* 将要读取文件头信息的文件的byte数组转换成string类型表示
*
* @param src
* 要读取文件头信息的文件的byte数组
* @return 文件头信息
*/
private static String bytesToHexString(byte[] src){
StringBuilder builder = new StringBuilder()
if (src == null || src.length <= 0){
return null
}
String hv
for (int i = 0 i < src.length i++){
// 以十六进制(基数 16)无符号整数形式返回一个整数参数的字符串表示形式,并转换为大写
hv = Integer.toHexString(src[i] & 0xFF).toUpperCase()
if (hv.length() < 2){
builder.append(0)
}
builder.append(hv)
}
System.out.println(builder.toString())
return builder.toString()
}
public static void main(String[] args)throws Exception {
final String fileType = getFileType("E:/Java编程思想读书笔记.docx")
System.out.println(fileType)
}
}
这个不应该靠判断数据去判断。应该另外增加一个叫content_type数据类型的数据库的列。然后在存入时写上数据类型,比如按MIME类型application/word之类,也可以枚举节约空间。
读取时根据该列的类型,给出后缀名。
也可以直接把原后缀名存在某列中。
也可以把原文件名存在某列中,而原文件放在某位置,并不入库。
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)