Android获取文件类型

Android获取文件类型,第1张

文件命名后缀如.jpg、.png等等一般即为文件类型,但有些时候不能保证文件后缀名一定有效,所以尝试将后缀转换成对应mime类型,方法如下:

如果转换结果返回为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之类,也可以枚举节约空间。

读取时根据该列的类型,给出后缀名。

也可以直接把原后缀名存在某列中。

也可以把原文件名存在某列中,而原文件放在某位置,并不入库。


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/tougao/12035392.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-05-20
下一篇2023-05-20

发表评论

登录后才能评论

评论列表(0条)

    保存