如何确定Android中文件的MIME类型?


176

假设我有完整的文件路径,例如:(/ sdcard / tlogo.png)。我想知道它的mime类型。

我为此创建了一个函数

public static String getMimeType(File file, Context context)    
{
    Uri uri = Uri.fromFile(file);
    ContentResolver cR = context.getContentResolver();
    MimeTypeMap mime = MimeTypeMap.getSingleton();
    String type = mime.getExtensionFromMimeType(cR.getType(uri));
    return type;
}

但是当我调用它时,它返回null。

File file = new File(filePath);
String fileType=CommonFunctions.getMimeType(file, context);

Answers:


323

首先,您应该考虑MimeTypeMap#getMimeTypeFromExtension()像这样调用:

// url = file path or whatever suitable URL you want.
public static String getMimeType(String url) {
    String type = null;
    String extension = MimeTypeMap.getFileExtensionFromUrl(url);
    if (extension != null) {
        type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
    }
    return type;
}

4
感谢我的工作,我的问题的另一个解决方案是:公共静态字符串getMimeType(String path,Context context){字符串扩展= p​​ath.substring(path.lastIndexOf(“。”)); 字符串mimeTypeMap = MimeTypeMap.getFileExtensionFromUrl(extension); 字符串mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(mimeTypeMap); return mimeType;}
Sushant Bhatnagar

13
如果文件没有好的扩展名?一个人可以制作一个.mp3文件,其中包含文本
throrin19 2012年

2
然后,您实际上需要打开文件并检查导入中的幻数(取决于c的格式)-但是,这还假定将txt文件重命名为mp3的人并不仅仅是ID3在他的文字开始让您更加困惑。
詹斯2012年

1
如果文件的扩展名未知。方法返回*/*吗?

3
如果您的文件路径为/ storage / emulated / 0 / Download / images(4).jpg,并且要求输入mimetype,则它将返回null。
Kalaiselvan '18

164

检测任何文件的MIME类型

public String getMimeType(Uri uri) {           
    String mimeType = null;
    if (uri.getScheme().equals(ContentResolver.SCHEME_CONTENT)) {
        ContentResolver cr = getAppContext().getContentResolver();
        mimeType = cr.getType(uri);
    } else {
        String fileExtension = MimeTypeMap.getFileExtensionFromUrl(uri
                .toString());
        mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(
                fileExtension.toLowerCase());
    }
    return mimeType;
}

5
这应该是正确的答案。toLowerCase()做到了。
拉贾特·萨克森纳

1
到目前为止,这是最好的答案,但是如果文件名带有特殊字符(在我的情况下为apostrophe!),它将失败。例如。文件名= Don't go baby.mp3。该PatternMatcher内部MimeTypeMap.getFileExtensionFromUrl()无法处理的文件名,并返回空字符串; null成为mimetype字符串!
sud007 '18

I. Shvedenenko对于我上面指出的问题,Worked 下面有一个类似的但无错误的解决方案。但这是正确方向的指针。
sud007 '18

此代码返回带有字符串“ / storage / emulated / 0 / dcim / screenshots / screenshot_20190319-123828_ubl digital app.jpg”的null。
阿卜杜勒

@Abdul这是因为该路径没有Uri方案。它应该以file://或开头content://
HB。

33

上面的MimeTypeMap解决方案在我的使用中返回null。这可行,并且更容易:

Uri uri = Uri.fromFile(file);
ContentResolver cR = context.getContentResolver();
String mime = cR.getType(uri);

19
此方法还可以返回null。
史密斯先生

根据URI,它看起来可以返回null与否。似乎没有人知道为什么;-(
Thomas Decaux 2014年

10
如果从android gallery中选择了媒体,则返回TYPE。如果从文件管理器中选择,则返回null。
Rahul Rastogi 2014年

我可以肯定地证实拉胡尔所说的话,只是测试了一下,他是对的
克里斯(Chris)

在某些设备中,如果我从图库中选择图像,它也会返回null
Ghanshyam Nayma

21

Jens回答者的优化版本,具有null安全性和后备类型。

@NonNull
static String getMimeType(@NonNull File file) {
    String type = null;
    final String url = file.toString();
    final String extension = MimeTypeMap.getFileExtensionFromUrl(url);
    if (extension != null) {
        type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension.toLowerCase());
    }
    if (type == null) {
        type = "image/*"; // fallback type. You might set it to */*
    }
    return type;
}

重要提示:getFileExtensionFromUrl()仅适用于小写字母


更新(19.03.2018)

奖励:以上方法作为不太复杂的Kotlin扩展函数

fun File.getMimeType(fallback: String = "image/*"): String {
    return MimeTypeMap.getFileExtensionFromUrl(toString())
            ?.run { MimeTypeMap.getSingleton().getMimeTypeFromExtension(toLowerCase()) }
            ?: fallback // You might set it to */*
}

这工作。虽然有点太冗长,但我很喜欢。
filthy_wizard

当然,这可以以较短的方式扭曲。对于Kotlin来说,可能只有两到三条生产线,但是福克斯却处于无效的toLoverCase地位。
Tobias

@filthy_wizard,为您添加了一个优化版本;-)
Tobias

对于我来说,使用getPath()方法而不是toString()更加清晰,并且仅使用kotlin属性访问语法提供路径。
Leonid

15
File file = new File(path, name);

    MimeTypeMap mime = MimeTypeMap.getSingleton();
    int index = file.getName().lastIndexOf('.')+1;
    String ext = file.getName().substring(index).toLowerCase();
    String type = mime.getMimeTypeFromExtension(ext);

    intent.setDataAndType(Uri.fromFile(file), type);
    try
    {
      context.startActivity(intent);
    }
    catch(ActivityNotFoundException ex)
    {
        ex.printStackTrace();

    }

1
对我来说很完美!但是如何使用FilenameUtils.getExetension(path)来获取文件扩展名呢?
彼得

7

请特别注意上面umerk44解决方案getMimeTypeFromExtension调用guessMimeTypeTypeFromExtension并且区分大小写。我对这个然后拿着细看花了一个下午的- getMimeTypeFromExtension将返回NULL如果你传递“JPG”,而这将返回“图像/ JPEG”如果你传递“JPG”。


5

这是我在Android应用中使用的解决方案:

public static String getMimeType(String url)
    {
        String extension = url.substring(url.lastIndexOf("."));
        String mimeTypeMap = MimeTypeMap.getFileExtensionFromUrl(extension);
        String mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(mimeTypeMap);
        return mimeType;
    }

1
网址?许多网址没有扩展名。例如,此页面没有扩展名。对于url,您应该使用新的URL(url).openConnection()。getRequestProperty(“ Content-type”);
barwnikk

5

有时Jeb和Jens的答案不起作用,并且返回null。在这种情况下,我使用关注解决方案。文件头通常包含类型签名。我阅读并与签名列表中的已知签名进行比较。

/**
 *
 * @param is InputStream on start of file. Otherwise signature can not be defined.
 * @return int id of signature or -1, if unknown signature was found. See SIGNATURE_ID_(type) constants to
 *      identify signature by its id.
 * @throws IOException in cases of read errors.
 */
public static int getSignatureIdFromHeader(InputStream is) throws IOException {
    // read signature from head of source and compare with known signatures
    int signatureId = -1;
    int sigCount = SIGNATURES.length;
    int[] byteArray = new int[MAX_SIGNATURE_LENGTH];
    StringBuilder builder = new StringBuilder();
    for (int i = 0; i < MAX_SIGNATURE_LENGTH; i++) {
        byteArray[i] = is.read();
        builder.append(Integer.toHexString(byteArray[i]));
    }
    if (DEBUG) {
        Log.d(TAG, "head bytes=" + builder.toString());
    }
    for (int i = 0; i < MAX_SIGNATURE_LENGTH; i++) {

        // check each bytes with known signatures
        int bytes = byteArray[i];
        int lastSigId = -1;
        int coincidences = 0;

        for (int j = 0; j < sigCount; j++) {
            int[] sig = SIGNATURES[j];

            if (DEBUG) {
                Log.d(TAG, "compare" + i + ": " + Integer.toHexString(bytes) + " with " + sig[i]);
            }
            if (bytes == sig[i]) {
                lastSigId = j;
                coincidences++;
            }
        }

        // signature is unknown
        if (coincidences == 0) {
            break;
        }
        // if first bytes of signature is known we check signature for full coincidence
        if (coincidences == 1) {
            int[] sig = SIGNATURES[lastSigId];
            int sigLength = sig.length;
            boolean isSigKnown = true;
            for (; i < MAX_SIGNATURE_LENGTH && i < sigLength; i++) {
                bytes = byteArray[i];
                if (bytes != sig[i]) {
                    isSigKnown = false;
                    break;
                }
            }
            if (isSigKnown) {
                signatureId = lastSigId;
            }
            break;
        }
    }
    return signatureId;
}

signatureId是签名数组中签名的索引。例如,

private static final int[] SIGNATURE_PNG = hexStringToIntArray("89504E470D0A1A0A");
private static final int[] SIGNATURE_JPEG = hexStringToIntArray("FFD8FF");
private static final int[] SIGNATURE_GIF = hexStringToIntArray("474946");

public static final int SIGNATURE_ID_JPEG = 0;
public static final int SIGNATURE_ID_PNG = 1;
public static final int SIGNATURE_ID_GIF = 2;
private static final int[][] SIGNATURES = new int[3][];

static {
    SIGNATURES[SIGNATURE_ID_JPEG] = SIGNATURE_JPEG;
    SIGNATURES[SIGNATURE_ID_PNG] = SIGNATURE_PNG;
    SIGNATURES[SIGNATURE_ID_GIF] = SIGNATURE_GIF;
}

现在我有了文件类型,即使文件的URI还没有。接下来,按文件类型获取mime类型。如果您不知道要获取哪种MIME类型,则可以在此表中找到合适的

它适用于许多文件类型。但是对于视频来说,它是行不通的,因为您需要使用已知的视频编解码器才能获得哑剧类型。要获取视频的MIME类型,请使用MediaMetadataRetriever


4

我尝试使用标准方法来确定mime类型,但是我无法使用 MimeTypeMap.getFileExtensionFromUrl(uri.getPath())。此方法返回了一个空字符串。因此,我提出了一个简单的解决方案来保留文件扩展名。

这是返回文件扩展名的方法:

private String getExtension(String fileName){
    char[] arrayOfFilename = fileName.toCharArray();
    for(int i = arrayOfFilename.length-1; i > 0; i--){
        if(arrayOfFilename[i] == '.'){
            return fileName.substring(i+1, fileName.length());
        }
    }
    return "";
}

保留了文件扩展名后,可以得到如下的mime类型:

public String getMimeType(File file) {
    String mimeType = "";
    String extension = getExtension(file.getName());
    if (MimeTypeMap.getSingleton().hasExtension(extension)) {
        mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
    }
    return mimeType;
}

伙计,这钉住了所有关于此问题的答案!所有旧版课程都存在相同的问题。问题与您提到的相同,MimeTypeMap.getFileExtensionFromUrl根据的内部实现,文件名返回了空字符串,其中包含无效字符PatternMatcher。这完全为我工作!喜欢它,到目前为止没有问题!
sud007 '18

@一世。Shvedenenko String extension = file.getName().split("\\.")[1]也可以工作,从而消除了getExtension()方法。
Shahood ul Hassan

但是问题是,如果在创建文件时使用了错误的扩展名怎么办?扩展应该是哑剧类型的唯一裁判吗?
Shahood ul Hassan


2

对于Xamarin Android(来自上面@HoaLe的回答)

public String getMimeType(Uri uri) {
    String mimeType = null;
    if (uri.Scheme.Equals(ContentResolver.SchemeContent))
    {
        ContentResolver cr = Application.Context.ContentResolver;
        mimeType = cr.GetType(uri);
    }
    else
    {
        String fileExtension = MimeTypeMap.GetFileExtensionFromUrl(uri.ToString());
        mimeType = MimeTypeMap.Singleton.GetMimeTypeFromExtension(
        fileExtension.ToLower());
    }
    return mimeType;
}

1
get file object....
File file = new File(filePath);

then....pass as a parameter to...

getMimeType(file);

...here is 


public String getMimeType(File file) {
        String mimetype = MimeTypeMap.getSingleton().getMimeTypeFromExtension(MimeTypeMap.getFileExtensionFromUrl(Uri.fromFile(file).toString()).toLowerCase());
        if (mimetype == null) {
            return "*/*";
        }
        return mimetype;///return the mimeType
    }

1

来自资产/文件时(请注意,MimeTypeMap中缺少一些情况)。

private String getMimeType(String path) {
    if (null == path) return "*/*";

    String extension = path;
    int lastDot = extension.lastIndexOf('.');
    if (lastDot != -1) {
        extension = extension.substring(lastDot + 1);
    }

    // Convert the URI string to lower case to ensure compatibility with MimeTypeMap (see CB-2185).
    extension = extension.toLowerCase(Locale.getDefault());
    if (extension.equals("3ga")) {
        return "audio/3gpp";
    } else if (extension.equals("js")) {
        return "text/javascript";
    } else if (extension.equals("woff")) {
        return "application/x-font-woff";
    } else {
        // TODO
        // anyting missing from the map (http://www.sitepoint.com/web-foundations/mime-types-complete-list/)
        // reference: http://grepcode.com/file/repo1.maven.org/maven2/com.google.okhttp/okhttp/20120626/libcore/net/MimeUtils.java#MimeUtils
    }

    return MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
}

在使用ContentResolver时

contentResolver.getType(uri)

当http / https请求时

    try {
        HttpURLConnection conn = httpClient.open(new URL(uri.toString()));
        conn.setDoInput(false);
        conn.setRequestMethod("HEAD");
        return conn.getHeaderField("Content-Type");
    } catch (IOException e) {
    }

1
// This will return the mimeType. 
// for eg. xyz.png it will return image/png. 
// here uri is the file that we were picked using intent from ext/internal storage.
private String getMimeType(Uri uri) {
   // This class provides applications access to the content model.  
   ContentResolver contentResolver = getContentResolver();

   // getType(Uri url)-Return the MIME type of the given content URL. 
   return contentResolver.getType(uri);
}

请提供一些解释。它将帮助初始读者适应她/他的代码,并帮助更多读者从您的答案中获得最大价值。
的Aurelien乙

感谢您的答复。我用解释对其进行编辑。
iamdhariot

1

我遇到了类似的问题。到目前为止,我知道不同名称的结果可能会有所不同,因此终于有了这个解决方案。

public String getMimeType(String filePath) {
    String type = null;
    String extension = null;
    int i = filePath.lastIndexOf('.');
    if (i > 0)
        extension = filePath.substring(i+1);
    if (extension != null)
        type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
    return type;
}  


0

来自本地文件的mime:

String url = file.getAbsolutePath();
FileNameMap fileNameMap = URLConnection.getFileNameMap();
String mime = fileNameMap.getContentTypeFor("file://"+url);

0

您有多种选择来获取文件扩展名:like:1- String filename = uri.getLastPathSegment();看到此链接

2-您也可以使用此代码

 filePath .substring(filePath.lastIndexOf(".")+1);

但这不是很好的选择。3-如果您具有文件的URI,则使用此代码

String[] projection = { MediaStore.MediaColumns.DATA,
MediaStore.MediaColumns.MIME_TYPE };

4-如果您有URL,请使用以下代码:

 public static String getMimeType(String url) {
String type = null;
String extension = MimeTypeMap.getFileExtensionFromUrl(url);
if (extension != null) { 


  type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
    }

return type;
}

享受你的代码:)


0
// new processing the mime type out of Uri which may return null in some cases
String mimeType = getContentResolver().getType(uri);
// old processing the mime type out of path using the extension part if new way returned null
if (mimeType == null){mimeType URLConnection.guessContentTypeFromName(path);}

0
public static String getFileType(Uri file)
{
    try
    {
        if (file.getScheme().equals(ContentResolver.SCHEME_CONTENT))
            return subStringFromLastMark(SystemMaster.getContentResolver().getType(file), "/");
        else
            return MimeTypeMap.getFileExtensionFromUrl(file.toString()).toLowerCase();
    }
    catch(Exception e)
    {
        return null;
    }
}

public static String getMimeType(Uri file)
{
    try
    {
        return MimeTypeMap.getSingleton().getMimeTypeFromExtension(getFileType(file));
    }
    catch(Exception e)
    {
        return null;
    }
}

public static String subStringFromLastMark(String str,String mark)
{
    int l = str.lastIndexOf(mark);
    int end = str.length();
    if(l == -1)
        return str;

    return str.substring(l + 1, end);
}

0

在我的案例路径中也返回了空值

/ storage / emulated / 0 / Music / 01-舞池上的鬼魂.mp3

围绕使用

val url = inUrl.replace(“”,“”)

所以方法看起来像

@JvmStatic
    fun getMimeType(inUrl: String?): String {
        if (inUrl == null) return ""

        val url = inUrl.replace(" ","")
        var type: String? = null

        val extension = MimeTypeMap.getFileExtensionFromUrl(url)
        if (extension != null) {
            type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension.toLowerCase())
        }

        if(type ==null){
            val cR = WifiTalkie.getApplicationContext().contentResolver
            type = cR.getType(Uri.parse(url))
        }

        if (type == null) {
            type = "*/*" // fallback method_type. You might set it to */*
        }
        return type
    }

结果返回成功结果:

音频/ mpeg

希望它可以帮助任何人


0

这里没有一个答案是完美的。这是一个结合了所有最佳答案的最佳元素的答案:

public final class FileUtil {

    // By default, Android doesn't provide support for JSON
    public static final String MIME_TYPE_JSON = "application/json";

    @Nullable
    public static String getMimeType(@NonNull Context context, @NonNull Uri uri) {

        String mimeType = null;
        if (uri.getScheme().equals(ContentResolver.SCHEME_CONTENT)) {
            ContentResolver cr = context.getContentResolver();
            mimeType = cr.getType(uri);
        } else {
            String fileExtension = getExtension(uri.toString());

            if(fileExtension == null){
                return null;
            }

            mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(
                    fileExtension.toLowerCase());

            if(mimeType == null){
                // Handle the misc file extensions
                return handleMiscFileExtensions(fileExtension);
            }
        }
        return mimeType;
    }

    @Nullable
    private static String getExtension(@Nullable String fileName){

        if(fileName == null || TextUtils.isEmpty(fileName)){
            return null;
        }

        char[] arrayOfFilename = fileName.toCharArray();
        for(int i = arrayOfFilename.length-1; i > 0; i--){
            if(arrayOfFilename[i] == '.'){
                return fileName.substring(i+1, fileName.length());
            }
        }
        return null;
    }

    @Nullable
    private static String handleMiscFileExtensions(@NonNull String extension){

        if(extension.equals("json")){
            return MIME_TYPE_JSON;
        }
        else{
            return null;
        }
    }
}

0
Intent myIntent = new Intent(android.content.Intent.ACTION_VIEW);
                        File file = new File(filePatch); 
                        Uri uris = Uri.fromFile(file);
                        String mimetype = null;
                        if 
(uris.getScheme().equals(ContentResolver.SCHEME_CONTENT)) {
                            ContentResolver cr = 
getApplicationContext().getContentResolver();
                            mimetype = cr.getType(uris);
                        } else {
                            String fileExtension = 
MimeTypeMap.getFileExtensionFromUrl(uris.toString());
mimetype =  MimeTypeMap.getSingleton().getMimeTypeFromExtension(fileExtension.toLowerCase());
                        }


0

我不知道为什么s和其他字符会MimeTypeMap.getFileExtensionFromUrl()出现问题space,这会返回"",但是我只是编写了此方法将文件名更改为允许的名称。它只是在玩Strings。但是,这是可行的。通过该方法,将space文件名中存在的s转换为所需的通孔(此处为“ x”),replaceAll(" ", "x")并将其他不合适的字符转换为合适的通孔URLEncoder。因此用法(根据问题和所选答案中显示的代码)应类似于getMimeType(reviseUrl(url))

private String reviseUrl(String url) {

        String revisedUrl = "";
        int fileNameBeginning = url.lastIndexOf("/");
        int fileNameEnding = url.lastIndexOf(".");

        String cutFileNameFromUrl = url.substring(fileNameBeginning + 1, fileNameEnding).replaceAll(" ", "x");

        revisedUrl = url.
                substring(0, fileNameBeginning + 1) +
                java.net.URLEncoder.encode(cutFileNameFromUrl) +
                url.substring(fileNameEnding, url.length());

        return revisedUrl;
    }
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.