我正在从 URL 下载 ePub 文件。
现在我想实现一种机制,如果用户尝试重新下载相同的文件,他应该得到警告/错误消息,并且不应该再次下载该文件。
为了实现这一点,我需要检查我的库中存在的文件的名称以及用户尝试下载的文件的名称。
但我只有这个下载链接,而不是文件名。
如何在下载前获取文件名以便与现有文件进行比较?
在 android 中,您可以使用guessFileName() 方法:
URLUtil.guessFileName(url, null, null)
或者,Java 中的简单解决方案可能是:
String fileName = url.substring(url.lastIndexOf('/') + 1);
(假设您的网址格式为http://xxxxxxxxxxxxx/filename.ext
:)
更新 2018 年 3 月 23 日
这个问题得到了很多点击,有人评论我的“简单”解决方案不适用于某些网址,所以我觉得有必要改进答案。
如果您想处理更复杂的 url 模式,我在下面提供了一个示例解决方案。它很快就变得相当复杂,我很确定我的解决方案仍然无法处理一些奇怪的情况,但它仍然存在:
public static String getFileNameFromURL(String url) {
if (url == null) {
return "";
}
try {
URL resource = new URL(url);
String host = resource.getHost();
if (host.length() > 0 && url.endsWith(host)) {
// handle ...example.com
return "";
}
}
catch(MalformedURLException e) {
return "";
}
int startIndex = url.lastIndexOf('/') + 1;
int length = url.length();
// find end index for ?
int lastQMPos = url.lastIndexOf('?');
if (lastQMPos == -1) {
lastQMPos = length;
}
// find end index for #
int lastHashPos = url.lastIndexOf('#');
if (lastHashPos == -1) {
lastHashPos = length;
}
// calculate the end index
int endIndex = Math.min(lastQMPos, lastHashPos);
return url.substring(startIndex, endIndex);
}
此方法可以处理这些类型的输入:
Input: "null" Output: ""
Input: "" Output: ""
Input: "file:///home/user/test.html" Output: "test.html"
Input: "file:///home/user/test.html?id=902" Output: "test.html"
Input: "file:///home/user/test.html#footer" Output: "test.html"
Input: "http://example.com" Output: ""
Input: "http://www.example.com" Output: ""
Input: "http://www.example.txt" Output: ""
Input: "http://example.com/" Output: ""
Input: "http://example.com/a/b/c/test.html" Output: "test.html"
Input: "http://example.com/a/b/c/test.html?param=value" Output: "test.html"
Input: "http://example.com/a/b/c/test.html#anchor" Output: "test.html"
Input: "http://example.com/a/b/c/test.html#anchor?param=value" Output: "test.html"
你可以在这里找到完整的源代码:https ://ideone.com/uFWxTL
尝试使用URLUtil.guessFileName(url, null, null)
例如。我认为这是最好的 Android 方式。
把事情简单化 :
/**
* This function will take an URL as input and return the file name.
* <p>Examples :</p>
* <ul>
* <li>http://example.com/a/b/c/test.txt -> test.txt</li>
* <li>http://example.com/ -> an empty string </li>
* <li>http://example.com/test.txt?param=value -> test.txt</li>
* <li>http://example.com/test.txt#anchor -> test.txt</li>
* </ul>
*
* @param url The input URL
* @return The URL file name
*/
public static String getFileNameFromUrl(URL url) {
String urlString = url.getFile();
return urlString.substring(urlString.lastIndexOf('/') + 1).split("\\?")[0].split("#")[0];
}
我认为使用 URL#getPath() 应该可以简化事情。
public static String getFileNameFromUrl(URL url) {
String urlPath = url.getPath();
return urlPath.substring(urlPath.lastIndexOf('/') + 1);
}
见,http://developer.android.com/reference/java/net/URL.html#getPath()
您实际上不必比较文件名。只需为具有绝对路径的文件创建File对象并检查文件是否存在。
protected boolean need2Download(String fileName) {
File basePath = new File(BOOK_STORE_PATH);
File fullPath = new File(basePath, fileName);
if (fullPath.exists())
return false;
return true;
}
protected void downloadFile(String url) {
String fileName = url.substring(url.lastIndexOf('/') + 1);
if (need2Download(fileName)) {
// download
}
}
FilenameUtils
从中获取文件名apache commons-io
URL url = URL(fileUrl)
String fileName = FilenameUtils.getName(url.getPath())
为此commons-io
在 build.gradle(app) 中添加依赖项
implementation "commons-io:commons-io:2.6"
它还从复杂的 URL 中提取文件名,如下所示
http://www.example.com/some/path/to/a/file.xml?foo=bar#test