All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cn.hutool.core.io.FileUtil Maven / Gradle / Ivy

Go to download

Hutool是一个小而全的Java工具类库,通过静态方法封装,降低相关API的学习成本,提高工作效率,使Java拥有函数式语言般的优雅,让Java语言也可以“甜甜的”。

There is a newer version: 5.8.34
Show newest version
package cn.hutool.core.io;

import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.io.file.FileCopier;
import cn.hutool.core.io.file.FileMode;
import cn.hutool.core.io.file.FileNameUtil;
import cn.hutool.core.io.file.FileReader;
import cn.hutool.core.io.file.FileReader.ReaderHandler;
import cn.hutool.core.io.file.FileWriter;
import cn.hutool.core.io.file.LineSeparator;
import cn.hutool.core.io.file.PathUtil;
import cn.hutool.core.io.file.Tailer;
import cn.hutool.core.io.resource.Resource;
import cn.hutool.core.io.resource.ResourceUtil;
import cn.hutool.core.io.unit.DataSizeUtil;
import cn.hutool.core.lang.Assert;
import cn.hutool.core.lang.Console;
import cn.hutool.core.thread.ThreadUtil;
import cn.hutool.core.util.ArrayUtil;
import cn.hutool.core.util.CharUtil;
import cn.hutool.core.util.CharsetUtil;
import cn.hutool.core.util.ClassUtil;
import cn.hutool.core.util.ReUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.core.util.URLUtil;
import cn.hutool.core.util.ZipUtil;

import java.io.*;
import java.net.URI;
import java.net.URL;
import java.net.URLConnection;
import java.nio.charset.Charset;
import java.nio.file.DirectoryNotEmptyException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Date;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.function.Consumer;
import java.util.function.Predicate;
import java.util.jar.JarFile;
import java.util.regex.Pattern;
import java.util.zip.CRC32;
import java.util.zip.Checksum;

/**
 * 文件工具类
 *
 * @author looly
 */
public class FileUtil extends PathUtil {

	/**
	 * Class文件扩展名
	 */
	public static final String CLASS_EXT = FileNameUtil.EXT_CLASS;
	/**
	 * Jar文件扩展名
	 */
	public static final String JAR_FILE_EXT = FileNameUtil.EXT_JAR;
	/**
	 * 在Jar中的路径jar的扩展名形式
	 */
	public static final String JAR_PATH_EXT = ".jar!";
	/**
	 * 当Path为文件形式时, path会加入一个表示文件的前缀
	 */
	public static final String PATH_FILE_PRE = URLUtil.FILE_URL_PREFIX;
	/**
	 * 文件路径分隔符
* 在Unix和Linux下 是{@code '/'}; 在Windows下是 {@code '\'} */ public static final String FILE_SEPARATOR = File.separator; /** * 多个PATH之间的分隔符
* 在Unix和Linux下 是{@code ':'}; 在Windows下是 {@code ';'} */ public static final String PATH_SEPARATOR = File.pathSeparator; /** * 绝对路径判断正则 */ private static final Pattern PATTERN_PATH_ABSOLUTE = Pattern.compile("^[a-zA-Z]:([/\\\\].*)?"); /** * 是否为Windows环境 * * @return 是否为Windows环境 * @since 3.0.9 */ public static boolean isWindows() { return FileNameUtil.WINDOWS_SEPARATOR == File.separatorChar; } /** * 列出指定路径下的目录和文件
* 给定的绝对路径不能是压缩包中的路径 * * @param path 目录绝对路径或者相对路径 * @return 文件列表(包含目录) */ public static File[] ls(String path) { if (path == null) { return null; } File file = file(path); if (file.isDirectory()) { return file.listFiles(); } throw new IORuntimeException(StrUtil.format("Path [{}] is not directory!", path)); } /** * 文件是否为空
* 目录:里面没有文件时为空 文件:文件大小为0时为空 * * @param file 文件 * @return 是否为空,当提供非目录时,返回false */ public static boolean isEmpty(File file) { if (null == file || false == file.exists()) { return true; } if (file.isDirectory()) { String[] subFiles = file.list(); return ArrayUtil.isEmpty(subFiles); } else if (file.isFile()) { return file.length() <= 0; } return false; } /** * 目录是否为空 * * @param file 目录 * @return 是否为空,当提供非目录时,返回false */ public static boolean isNotEmpty(File file) { return false == isEmpty(file); } /** * 目录是否为空 * * @param dir 目录 * @return 是否为空 */ public static boolean isDirEmpty(File dir) { return isDirEmpty(dir.toPath()); } /** * 递归遍历目录以及子目录中的所有文件
* 如果提供file为文件,直接返回过滤结果 * * @param path 当前遍历文件或目录的路径 * @param fileFilter 文件过滤规则对象,选择要保留的文件,只对文件有效,不过滤目录 * @return 文件列表 * @since 3.2.0 */ public static List loopFiles(String path, FileFilter fileFilter) { return loopFiles(file(path), fileFilter); } /** * 递归遍历目录以及子目录中的所有文件
* 如果提供file为文件,直接返回过滤结果 * * @param file 当前遍历文件或目录 * @param fileFilter 文件过滤规则对象,选择要保留的文件,只对文件有效,不过滤目录 * @return 文件列表 */ public static List loopFiles(File file, FileFilter fileFilter) { return loopFiles(file, -1, fileFilter); } /** * 递归遍历目录并处理目录下的文件,可以处理目录或文件: *
    *
  • 非目录则直接调用{@link Consumer}处理
  • *
  • 目录则递归调用此方法处理
  • *
* * @param file 文件或目录,文件直接处理 * @param consumer 文件处理器,只会处理文件 * @since 5.5.2 */ public static void walkFiles(File file, Consumer consumer) { if (file.isDirectory()) { final File[] subFiles = file.listFiles(); if (ArrayUtil.isNotEmpty(subFiles)) { for (File tmp : subFiles) { walkFiles(tmp, consumer); } } } else { consumer.accept(file); } } /** * 递归遍历目录以及子目录中的所有文件
* 如果提供file为文件,直接返回过滤结果 * * @param file 当前遍历文件或目录 * @param maxDepth 遍历最大深度,-1表示遍历到没有目录为止 * @param fileFilter 文件过滤规则对象,选择要保留的文件,只对文件有效,不过滤目录,null表示接收全部文件 * @return 文件列表 * @since 4.6.3 */ public static List loopFiles(File file, int maxDepth, FileFilter fileFilter) { return loopFiles(file.toPath(), maxDepth, fileFilter); } /** * 递归遍历目录以及子目录中的所有文件
* 如果用户传入相对路径,则是相对classpath的路径
* 如:"test/aaa"表示"${classpath}/test/aaa" * * @param path 相对ClassPath的目录或者绝对路径目录 * @return 文件列表 * @since 3.2.0 */ public static List loopFiles(String path) { return loopFiles(file(path)); } /** * 递归遍历目录以及子目录中的所有文件 * * @param file 当前遍历文件 * @return 文件列表 */ public static List loopFiles(File file) { return loopFiles(file, null); } /** * 获得指定目录下所有文件
* 不会扫描子目录
* 如果用户传入相对路径,则是相对classpath的路径
* 如:"test/aaa"表示"${classpath}/test/aaa" * * @param path 相对ClassPath的目录或者绝对路径目录 * @return 文件路径列表(如果是jar中的文件,则给定类似.jar!/xxx/xxx的路径) * @throws IORuntimeException IO异常 */ public static List listFileNames(String path) throws IORuntimeException { if (path == null) { return new ArrayList<>(0); } int index = path.lastIndexOf(FileUtil.JAR_PATH_EXT); if (index < 0) { // 普通目录 final List paths = new ArrayList<>(); final File[] files = ls(path); for (File file : files) { if (file.isFile()) { paths.add(file.getName()); } } return paths; } // jar文件 path = getAbsolutePath(path); // jar文件中的路径 index = index + FileUtil.JAR_FILE_EXT.length(); JarFile jarFile = null; try { jarFile = new JarFile(path.substring(0, index)); // 防止出现jar!/cn/hutool/这类路径导致文件找不到 return ZipUtil.listFileNames(jarFile, StrUtil.removePrefix(path.substring(index + 1), "/")); } catch (IOException e) { throw new IORuntimeException(StrUtil.format("Can not read file path of [{}]", path), e); } finally { IoUtil.close(jarFile); } } /** * 创建File对象,相当于调用new File(),不做任何处理 * * @param path 文件路径,相对路径表示相对项目路径 * @return File * @since 4.1.4 */ public static File newFile(String path) { return new File(path); } /** * 创建File对象,自动识别相对或绝对路径,相对路径将自动从ClassPath下寻找 * * @param path 相对ClassPath的目录或者绝对路径目录 * @return File */ public static File file(String path) { if (null == path) { return null; } return new File(getAbsolutePath(path)); } /** * 创建File对象
* 此方法会检查slip漏洞,漏洞说明见http://blog.nsfocus.net/zip-slip-2/ * * @param parent 父目录 * @param path 文件路径 * @return File */ public static File file(String parent, String path) { return file(new File(parent), path); } /** * 创建File对象
* 根据的路径构建文件,在Win下直接构建,在Linux下拆分路径单独构建 * 此方法会检查slip漏洞,漏洞说明见http://blog.nsfocus.net/zip-slip-2/ * * @param parent 父文件对象 * @param path 文件路径 * @return File */ public static File file(File parent, String path) { if (StrUtil.isBlank(path)) { throw new NullPointerException("File path is blank!"); } return checkSlip(parent, buildFile(parent, path)); } /** * 通过多层目录参数创建文件
* 此方法会检查slip漏洞,漏洞说明见http://blog.nsfocus.net/zip-slip-2/ * * @param directory 父目录 * @param names 元素名(多层目录名),由外到内依次传入 * @return the file 文件 * @since 4.0.6 */ public static File file(File directory, String... names) { Assert.notNull(directory, "directory must not be null"); if (ArrayUtil.isEmpty(names)) { return directory; } File file = directory; for (String name : names) { if (null != name) { file = file(file, name); } } return file; } /** * 通过多层目录创建文件 *

* 元素名(多层目录名) * * @param names 多层文件的文件名,由外到内依次传入 * @return the file 文件 * @since 4.0.6 */ public static File file(String... names) { if (ArrayUtil.isEmpty(names)) { return null; } File file = null; for (String name : names) { if (file == null) { file = file(name); } else { file = file(file, name); } } return file; } /** * 创建File对象 * * @param uri 文件URI * @return File */ public static File file(URI uri) { if (uri == null) { throw new NullPointerException("File uri is null!"); } return new File(uri); } /** * 创建File对象 * * @param url 文件URL * @return File */ public static File file(URL url) { return new File(URLUtil.toURI(url)); } /** * 获取临时文件路径(绝对路径) * * @return 临时文件路径 * @since 4.0.6 */ public static String getTmpDirPath() { return System.getProperty("java.io.tmpdir"); } /** * 获取临时文件目录 * * @return 临时文件目录 * @since 4.0.6 */ public static File getTmpDir() { return file(getTmpDirPath()); } /** * 获取用户路径(绝对路径) * * @return 用户路径 * @since 4.0.6 */ public static String getUserHomePath() { return System.getProperty("user.home"); } /** * 获取用户目录 * * @return 用户目录 * @since 4.0.6 */ public static File getUserHomeDir() { return file(getUserHomePath()); } /** * 判断文件是否存在,如果path为null,则返回false * * @param path 文件路径 * @return 如果存在返回true */ public static boolean exist(String path) { return (null != path) && file(path).exists(); } /** * 判断文件是否存在,如果file为null,则返回false * * @param file 文件 * @return 如果存在返回true */ public static boolean exist(File file) { return (null != file) && file.exists(); } /** * 是否存在匹配文件 * * @param directory 文件夹路径 * @param regexp 文件夹中所包含文件名的正则表达式 * @return 如果存在匹配文件返回true */ public static boolean exist(String directory, String regexp) { final File file = new File(directory); if (false == file.exists()) { return false; } final String[] fileList = file.list(); if (fileList == null) { return false; } for (String fileName : fileList) { if (fileName.matches(regexp)) { return true; } } return false; } /** * 指定文件最后修改时间 * * @param file 文件 * @return 最后修改时间 */ public static Date lastModifiedTime(File file) { if (false == exist(file)) { return null; } return new Date(file.lastModified()); } /** * 指定路径文件最后修改时间 * * @param path 绝对路径 * @return 最后修改时间 */ public static Date lastModifiedTime(String path) { return lastModifiedTime(new File(path)); } /** * 计算目录或文件的总大小
* 当给定对象为文件时,直接调用 {@link File#length()}
* 当给定对象为目录时,遍历目录下的所有文件和目录,递归计算其大小,求和返回
* 此方法不包括目录本身的占用空间大小。 * * @param file 目录或文件,null或者文件不存在返回0 * @return 总大小,bytes长度 */ public static long size(File file) { return size(file, false); } /** * 计算目录或文件的总大小
* 当给定对象为文件时,直接调用 {@link File#length()}
* 当给定对象为目录时,遍历目录下的所有文件和目录,递归计算其大小,求和返回 * * @param file 目录或文件,null或者文件不存在返回0 * @param includeDirSize 是否包括每层目录本身的大小 * @return 总大小,bytes长度 * @since 5.7.21 */ public static long size(File file, boolean includeDirSize) { if (null == file || false == file.exists() || isSymlink(file)) { return 0; } if (file.isDirectory()) { long size = includeDirSize ? file.length() : 0; File[] subFiles = file.listFiles(); if (ArrayUtil.isEmpty(subFiles)) { return 0L;// empty directory } for (File subFile : subFiles) { size += size(subFile, includeDirSize); } return size; } else { return file.length(); } } /** * 计算文件的总行数
* 参考:https://stackoverflow.com/questions/453018/number-of-lines-in-a-file-in-java * * @param file 文件 * @return 该文件总行数 * @since 5.7.22 */ public static int getTotalLines(File file) { return getTotalLines(file, 1024); } /** * 计算文件的总行数
* 参考:https://stackoverflow.com/questions/453018/number-of-lines-in-a-file-in-java * * @param file 文件 * @param bufferSize 缓存大小,小于1则使用默认的1024 * @return 该文件总行数 * @since 5.8.28 */ public static int getTotalLines(File file, int bufferSize) { if (false == isFile(file)) { throw new IORuntimeException("Input must be a File"); } if (bufferSize < 1) { bufferSize = 1024; } try (InputStream is = getInputStream(file)) { byte[] c = new byte[bufferSize]; int readChars = is.read(c); if (readChars == -1) { // 空文件,返回0 return 0; } // 起始行为1 // 如果只有一行,无换行符,则读取结束后返回1 // 如果多行,最后一行无换行符,最后一行需要单独计数 // 如果多行,最后一行有换行符,则空行算作一行 int count = 1; while (readChars == bufferSize) { for (int i = 0; i < bufferSize; i++) { if (c[i] == CharUtil.LF) { ++count; } } readChars = is.read(c); } // count remaining characters while (readChars != -1) { for (int i = 0; i < readChars; i++) { if (c[i] == CharUtil.LF) { ++count; } } readChars = is.read(c); } return count; } catch (IOException e) { throw new IORuntimeException(e); } } /** * 给定文件或目录的最后修改时间是否晚于给定时间 * * @param file 文件或目录 * @param reference 参照文件 * @return 是否晚于给定时间 */ public static boolean newerThan(File file, File reference) { if (null == reference || false == reference.exists()) { return true;// 文件一定比一个不存在的文件新 } return newerThan(file, reference.lastModified()); } /** * 给定文件或目录的最后修改时间是否晚于给定时间 * * @param file 文件或目录 * @param timeMillis 做为对比的时间 * @return 是否晚于给定时间 */ public static boolean newerThan(File file, long timeMillis) { if (null == file || false == file.exists()) { return false;// 不存在的文件一定比任何时间旧 } return file.lastModified() > timeMillis; } /** * 创建文件及其父目录,如果这个文件存在,直接返回这个文件
* 此方法不对File对象类型做判断,如果File不存在,无法判断其类型 * * @param path 相对ClassPath的目录或者绝对路径目录,使用POSIX风格 * @return 文件,若路径为null,返回null * @throws IORuntimeException IO异常 */ public static File touch(String path) throws IORuntimeException { if (path == null) { return null; } return touch(file(path)); } /** * 创建文件及其父目录,如果这个文件存在,直接返回这个文件
* 此方法不对File对象类型做判断,如果File不存在,无法判断其类型 * * @param file 文件对象 * @return 文件,若路径为null,返回null * @throws IORuntimeException IO异常 */ public static File touch(File file) throws IORuntimeException { if (null == file) { return null; } if (false == file.exists()) { mkParentDirs(file); try { //noinspection ResultOfMethodCallIgnored file.createNewFile(); } catch (Exception e) { throw new IORuntimeException(e); } } return file; } /** * 创建文件及其父目录,如果这个文件存在,直接返回这个文件
* 此方法不对File对象类型做判断,如果File不存在,无法判断其类型 * * @param parent 父文件对象 * @param path 文件路径 * @return File * @throws IORuntimeException IO异常 */ public static File touch(File parent, String path) throws IORuntimeException { return touch(file(parent, path)); } /** * 创建文件及其父目录,如果这个文件存在,直接返回这个文件
* 此方法不对File对象类型做判断,如果File不存在,无法判断其类型 * * @param parent 父文件对象 * @param path 文件路径 * @return File * @throws IORuntimeException IO异常 */ public static File touch(String parent, String path) throws IORuntimeException { return touch(file(parent, path)); } /** * 创建所给文件或目录的父目录 * * @param file 文件或目录 * @return 父目录 */ public static File mkParentDirs(File file) { if (null == file) { return null; } return mkdir(getParent(file, 1)); } /** * 创建父文件夹,如果存在直接返回此文件夹 * * @param path 文件夹路径,使用POSIX格式,无论哪个平台 * @return 创建的目录 */ public static File mkParentDirs(String path) { if (path == null) { return null; } return mkParentDirs(file(path)); } /** * 删除文件或者文件夹
* 路径如果为相对路径,会转换为ClassPath路径! 注意:删除文件夹时不会判断文件夹是否为空,如果不空则递归删除子文件或文件夹
* 某个文件删除失败会终止删除操作 * * @param fullFileOrDirPath 文件或者目录的路径 * @return 成功与否 * @throws IORuntimeException IO异常 */ public static boolean del(String fullFileOrDirPath) throws IORuntimeException { return del(file(fullFileOrDirPath)); } /** * 删除文件或者文件夹
* 注意:删除文件夹时不会判断文件夹是否为空,如果不空则递归删除子文件或文件夹
* 某个文件删除失败会终止删除操作 * *

* 从5.7.6开始,删除文件使用{@link Files#delete(Path)}代替 {@link File#delete()}
* 因为前者遇到文件被占用等原因时,抛出异常,而非返回false,异常会指明具体的失败原因。 *

* * @param file 文件对象 * @return 成功与否 * @throws IORuntimeException IO异常 * @see Files#delete(Path) */ public static boolean del(File file) throws IORuntimeException { if (file == null || false == file.exists()) { // 如果文件不存在或已被删除,此处返回true表示删除成功 return true; } if (file.isDirectory()) { // 清空目录下所有文件和目录 boolean isOk = clean(file); if (false == isOk) { return false; } } // 删除文件或清空后的目录 final Path path = file.toPath(); try { delFile(path); } catch (DirectoryNotEmptyException e) { // 遍历清空目录没有成功,此时补充删除一次(可能存在部分软链) del(path); } catch (IOException e) { throw new IORuntimeException(e); } return true; } /** * 清空文件夹
* 注意:清空文件夹时不会判断文件夹是否为空,如果不空则递归删除子文件或文件夹
* 某个文件删除失败会终止删除操作 * * @param dirPath 文件夹路径 * @return 成功与否 * @throws IORuntimeException IO异常 * @since 4.0.8 */ public static boolean clean(String dirPath) throws IORuntimeException { return clean(file(dirPath)); } /** * 清空文件夹
* 注意:清空文件夹时不会判断文件夹是否为空,如果不空则递归删除子文件或文件夹
* 某个文件删除失败会终止删除操作 * * @param directory 文件夹 * @return 成功与否 * @throws IORuntimeException IO异常 * @since 3.0.6 */ public static boolean clean(File directory) throws IORuntimeException { if (directory == null || directory.exists() == false || false == directory.isDirectory()) { return true; } final File[] files = directory.listFiles(); if (null != files) { for (File childFile : files) { if (false == del(childFile)) { // 删除一个出错则本次删除任务失败 return false; } } } return true; } /** * 清理空文件夹
* 此方法用于递归删除空的文件夹,不删除文件
* 如果传入的文件夹本身就是空的,删除这个文件夹 * * @param directory 文件夹 * @return 成功与否 * @throws IORuntimeException IO异常 * @since 4.5.5 */ public static boolean cleanEmpty(File directory) throws IORuntimeException { if (directory == null || false == directory.exists() || false == directory.isDirectory()) { return true; } final File[] files = directory.listFiles(); if (ArrayUtil.isEmpty(files)) { // 空文件夹则删除之 return directory.delete(); } for (File childFile : files) { cleanEmpty(childFile); } return true; } /** * 创建文件夹,如果存在直接返回此文件夹
* 此方法不对File对象类型做判断,如果File不存在,无法判断其类型 * * @param dirPath 文件夹路径,使用POSIX格式,无论哪个平台 * @return 创建的目录 */ public static File mkdir(String dirPath) { if (dirPath == null) { return null; } final File dir = file(dirPath); return mkdir(dir); } /** * 创建文件夹,会递归自动创建其不存在的父文件夹,如果存在直接返回此文件夹
* 此方法不对File对象类型做判断,如果File不存在,无法判断其类型
* * @param dir 目录 * @return 创建的目录 */ public static File mkdir(File dir) { if (dir == null) { return null; } if (false == dir.exists()) { mkdirsSafely(dir, 5, 1); } return dir; } /** * 安全地级联创建目录 (确保并发环境下能创建成功) * *
	 *     并发环境下,假设 test 目录不存在,如果线程A mkdirs "test/A" 目录,线程B mkdirs "test/B"目录,
	 *     其中一个线程可能会失败,进而导致以下代码抛出 FileNotFoundException 异常
	 *
	 *     file.getParentFile().mkdirs(); // 父目录正在被另一个线程创建中,返回 false
	 *     file.createNewFile(); // 抛出 IO 异常,因为该线程无法感知到父目录已被创建
	 * 
* * @param dir 待创建的目录 * @param tryCount 最大尝试次数 * @param sleepMillis 线程等待的毫秒数 * @return true表示创建成功,false表示创建失败 * @author z8g * @since 5.7.21 */ public static boolean mkdirsSafely(File dir, int tryCount, long sleepMillis) { if (dir == null) { return false; } if (dir.isDirectory()) { return true; } for (int i = 1; i <= tryCount; i++) { // 高并发场景下,可以看到 i 处于 1 ~ 3 之间 // 如果文件已存在,也会返回 false,所以该值不能作为是否能创建的依据,因此不对其进行处理 //noinspection ResultOfMethodCallIgnored dir.mkdirs(); if (dir.exists()) { return true; } ThreadUtil.sleep(sleepMillis); } return dir.exists(); } /** * 创建临时文件
* 创建后的文件名为 prefix[Random.tmp * * @param dir 临时文件创建的所在目录 * @return 临时文件 * @throws IORuntimeException IO异常 */ public static File createTempFile(File dir) throws IORuntimeException { return createTempFile("hutool", null, dir, true); } /** * 在默认临时文件目录下创建临时文件,创建后的文件名为 prefix[Random].tmp。 * 默认临时文件目录由系统属性 {@code java.io.tmpdir} 指定。 * 在 UNIX 系统上,此属性的默认值通常是 {@code "tmp"} 或 {@code "vartmp"}; * 在 Microsoft Windows 系统上,它通常是 {@code "C:\\WINNT\\TEMP"}。 * 调用 Java 虚拟机时,可以为该系统属性赋予不同的值,但不保证对该属性的编程更改对该方法使用的临时目录有任何影响。 * * @return 临时文件 * @throws IORuntimeException IO异常 * @since 5.7.22 */ public static File createTempFile() throws IORuntimeException { return createTempFile("hutool", null, null, true); } /** * 在默认临时文件目录下创建临时文件,创建后的文件名为 prefix[Random].suffix。 * 默认临时文件目录由系统属性 {@code java.io.tmpdir} 指定。 * 在 UNIX 系统上,此属性的默认值通常是 {@code "tmp"} 或 {@code "vartmp"}; * 在 Microsoft Windows 系统上,它通常是 {@code "C:\\WINNT\\TEMP"}。 * 调用 Java 虚拟机时,可以为该系统属性赋予不同的值,但不保证对该属性的编程更改对该方法使用的临时目录有任何影响。 * * @param suffix 后缀,如果null则使用默认.tmp * @param isReCreat 是否重新创建文件(删掉原来的,创建新的) * @return 临时文件 * @throws IORuntimeException IO异常 * @since 5.7.22 */ public static File createTempFile(String suffix, boolean isReCreat) throws IORuntimeException { return createTempFile("hutool", suffix, null, isReCreat); } /** * 在默认临时文件目录下创建临时文件,创建后的文件名为 prefix[Random].suffix。 * 默认临时文件目录由系统属性 {@code java.io.tmpdir} 指定。 * 在 UNIX 系统上,此属性的默认值通常是 {@code "tmp"} 或 {@code "vartmp"}; * 在 Microsoft Windows 系统上,它通常是 {@code "C:\\WINNT\\TEMP"}。 * 调用 Java 虚拟机时,可以为该系统属性赋予不同的值,但不保证对该属性的编程更改对该方法使用的临时目录有任何影响。 * * @param prefix 前缀,至少3个字符 * @param suffix 后缀,如果null则使用默认.tmp * @param isReCreat 是否重新创建文件(删掉原来的,创建新的) * @return 临时文件 * @throws IORuntimeException IO异常 * @since 5.7.22 */ public static File createTempFile(String prefix, String suffix, boolean isReCreat) throws IORuntimeException { return createTempFile(prefix, suffix, null, isReCreat); } /** * 创建临时文件
* 创建后的文件名为 prefix[Random].tmp * * @param dir 临时文件创建的所在目录 * @param isReCreat 是否重新创建文件(删掉原来的,创建新的) * @return 临时文件 * @throws IORuntimeException IO异常 */ public static File createTempFile(File dir, boolean isReCreat) throws IORuntimeException { return createTempFile("hutool", null, dir, isReCreat); } /** * 创建临时文件
* 创建后的文件名为 prefix[Random].suffix From com.jodd.io.FileUtil * * @param prefix 前缀,至少3个字符 * @param suffix 后缀,如果null则使用默认.tmp * @param dir 临时文件创建的所在目录 * @param isReCreat 是否重新创建文件(删掉原来的,创建新的) * @return 临时文件 * @throws IORuntimeException IO异常 */ public static File createTempFile(String prefix, String suffix, File dir, boolean isReCreat) throws IORuntimeException { int exceptionsCount = 0; while (true) { try { // https://github.com/dromara/hutool/issues/3103 //File file = File.createTempFile(prefix, suffix, mkdir(dir)).getCanonicalFile(); final File file = PathUtil.createTempFile(prefix, suffix, null == dir ? null : dir.toPath()).toFile().getCanonicalFile(); if (isReCreat) { //noinspection ResultOfMethodCallIgnored file.delete(); //noinspection ResultOfMethodCallIgnored file.createNewFile(); } return file; } catch (IOException ioex) { // fixes java.io.WinNTFileSystem.createFileExclusively access denied if (++exceptionsCount >= 50) { throw new IORuntimeException(ioex); } } } } /** * 通过JDK7+的 Files#copy(Path, Path, CopyOption...) 方法拷贝文件 * * @param src 源文件路径 * @param dest 目标文件或目录路径,如果为目录使用与源文件相同的文件名 * @param options {@link StandardCopyOption} * @return File * @throws IORuntimeException IO异常 */ public static File copyFile(String src, String dest, StandardCopyOption... options) throws IORuntimeException { Assert.notBlank(src, "Source File path is blank !"); Assert.notBlank(dest, "Destination File path is blank !"); return copyFile(Paths.get(src), Paths.get(dest), options).toFile(); } /** * 通过JDK7+的 Files#copy(InputStream, Path, CopyOption...) 方法拷贝文件 * * @param src 源文件 * @param dest 目标文件或目录,如果为目录使用与源文件相同的文件名 * @param options {@link StandardCopyOption} * @return 目标文件 * @throws IORuntimeException IO异常 * @since 5.8.27 */ public static File copyFile(Resource src, File dest, StandardCopyOption... options) throws IORuntimeException { // check Assert.notNull(src, "Source File is null !"); Assert.notNull(dest, "Destination File or directiory is null !"); return copyFile(src, dest.toPath(), options).toFile(); } /** * 通过JDK7+的 Files#copy(InputStream, Path, CopyOption...) 方法拷贝文件 * * @param src 源文件流,使用后不关闭 * @param dest 目标文件,不存在自动创建 * @param options {@link StandardCopyOption} * @return 目标文件 * @throws IORuntimeException IO异常 * @since 5.8.27 */ public static File copyFile(InputStream src, File dest, StandardCopyOption... options) throws IORuntimeException { // check Assert.notNull(src, "Source File is null !"); Assert.notNull(dest, "Destination File or directiory is null !"); return copyFile(src, dest.toPath(), options).toFile(); } /** * 通过JDK7+的 Files#copy(Path, Path, CopyOption...) 方法拷贝文件 * * @param src 源文件 * @param dest 目标文件或目录,如果为目录使用与源文件相同的文件名 * @param options {@link StandardCopyOption} * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File copyFile(File src, File dest, StandardCopyOption... options) throws IORuntimeException { // check Assert.notNull(src, "Source File is null !"); if (false == src.exists()) { throw new IORuntimeException("File not exist: " + src); } Assert.notNull(dest, "Destination File or directiory is null !"); if (equals(src, dest)) { throw new IORuntimeException("Files '{}' and '{}' are equal", src, dest); } return copyFile(src.toPath(), dest.toPath(), options).toFile(); } /** * 复制文件或目录
* 如果目标文件为目录,则将源文件以相同文件名拷贝到目标目录 * * @param srcPath 源文件或目录 * @param destPath 目标文件或目录,目标不存在会自动创建(目录、文件都创建) * @param isOverride 是否覆盖目标文件 * @return 目标目录或文件 * @throws IORuntimeException IO异常 */ public static File copy(String srcPath, String destPath, boolean isOverride) throws IORuntimeException { return copy(file(srcPath), file(destPath), isOverride); } /** * 复制文件或目录
* 情况如下: * *
	 * 1、src和dest都为目录,则将src目录及其目录下所有文件目录拷贝到dest下
	 * 2、src和dest都为文件,直接复制,名字为dest
	 * 3、src为文件,dest为目录,将src拷贝到dest目录下
	 * 
* * @param src 源文件 * @param dest 目标文件或目录,目标不存在会自动创建(目录、文件都创建) * @param isOverride 是否覆盖目标文件 * @return 目标目录或文件 * @throws IORuntimeException IO异常 */ public static File copy(File src, File dest, boolean isOverride) throws IORuntimeException { return FileCopier.create(src, dest).setOverride(isOverride).copy(); } /** * 复制文件或目录
* 情况如下: * *
	 * 1、src和dest都为目录,则将src下所有文件目录拷贝到dest下
	 * 2、src和dest都为文件,直接复制,名字为dest
	 * 3、src为文件,dest为目录,将src拷贝到dest目录下
	 * 
* * @param src 源文件 * @param dest 目标文件或目录,目标不存在会自动创建(目录、文件都创建) * @param isOverride 是否覆盖目标文件 * @return 目标目录或文件 * @throws IORuntimeException IO异常 */ public static File copyContent(File src, File dest, boolean isOverride) throws IORuntimeException { return FileCopier.create(src, dest).setCopyContentIfDir(true).setOverride(isOverride).copy(); } /** * 复制文件或目录
* 情况如下: * *
	 * 1、src和dest都为目录,则将src下所有文件(不包括子目录)拷贝到dest下
	 * 2、src和dest都为文件,直接复制,名字为dest
	 * 3、src为文件,dest为目录,将src拷贝到dest目录下
	 * 
* * @param src 源文件 * @param dest 目标文件或目录,目标不存在会自动创建(目录、文件都创建) * @param isOverride 是否覆盖目标文件 * @return 目标目录或文件 * @throws IORuntimeException IO异常 * @since 4.1.5 */ public static File copyFilesFromDir(File src, File dest, boolean isOverride) throws IORuntimeException { return FileCopier.create(src, dest).setCopyContentIfDir(true).setOnlyCopyFile(true).setOverride(isOverride).copy(); } /** * 移动文件或者目录 * * @param src 源文件或者目录 * @param target 目标文件或者目录 * @param isOverride 是否覆盖目标,只有目标为文件才覆盖 * @throws IORuntimeException IO异常 * @see PathUtil#move(Path, Path, boolean) */ public static void move(File src, File target, boolean isOverride) throws IORuntimeException { Assert.notNull(src, "Src file must be not null!"); Assert.notNull(target, "target file must be not null!"); move(src.toPath(), target.toPath(), isOverride); } /** * 移动文件或者目录 * * @param src 源文件或者目录 * @param target 目标文件或者目录 * @param isOverride 是否覆盖目标,只有目标为文件才覆盖 * @throws IORuntimeException IO异常 * @see PathUtil#moveContent(Path, Path, boolean) * @since 5.7.9 */ public static void moveContent(File src, File target, boolean isOverride) throws IORuntimeException { Assert.notNull(src, "Src file must be not null!"); Assert.notNull(target, "target file must be not null!"); moveContent(src.toPath(), target.toPath(), isOverride); } /** * 修改文件或目录的文件名,不变更路径,只是简单修改文件名,不保留扩展名。
* *
	 * FileUtil.rename(file, "aaa.png", true) xx/xx.png =》xx/aaa.png
	 * 
* * @param file 被修改的文件 * @param newName 新的文件名,如需扩展名,需自行在此参数加上,原文件名的扩展名不会被保留 * @param isOverride 是否覆盖目标文件 * @return 目标文件 * @since 5.3.6 */ public static File rename(File file, String newName, boolean isOverride) { return rename(file, newName, false, isOverride); } /** * 修改文件或目录的文件名,不变更路径,只是简单修改文件名
* 重命名有两种模式:
* 1、isRetainExt为true时,保留原扩展名: * *
	 * FileUtil.rename(file, "aaa", true) xx/xx.png =》xx/aaa.png
	 * 
* *

* 2、isRetainExt为false时,不保留原扩展名,需要在newName中 * *

	 * FileUtil.rename(file, "aaa.jpg", false) xx/xx.png =》xx/aaa.jpg
	 * 
* * @param file 被修改的文件 * @param newName 新的文件名,可选是否包括扩展名 * @param isRetainExt 是否保留原文件的扩展名,如果保留,则newName不需要加扩展名 * @param isOverride 是否覆盖目标文件 * @return 目标文件 * @see PathUtil#rename(Path, String, boolean) * @since 3.0.9 */ public static File rename(File file, String newName, boolean isRetainExt, boolean isOverride) { if (isRetainExt) { final String extName = FileUtil.extName(file); if (StrUtil.isNotBlank(extName)) { newName = newName.concat(".").concat(extName); } } return rename(file.toPath(), newName, isOverride).toFile(); } /** * 获取规范的绝对路径 * * @param file 文件 * @return 规范绝对路径,如果传入file为null,返回null * @since 4.1.4 */ public static String getCanonicalPath(File file) { if (null == file) { return null; } try { return file.getCanonicalPath(); } catch (IOException e) { throw new IORuntimeException(e); } } /** * 获取绝对路径
* 此方法不会判定给定路径是否有效(文件或目录存在) * * @param path 相对路径 * @param baseClass 相对路径所相对的类 * @return 绝对路径 */ public static String getAbsolutePath(String path, Class baseClass) { String normalPath; if (path == null) { normalPath = StrUtil.EMPTY; } else { normalPath = normalize(path); if (isAbsolutePath(normalPath)) { // 给定的路径已经是绝对路径了 return normalPath; } } // 相对于ClassPath路径 final URL url = ResourceUtil.getResource(normalPath, baseClass); if (null != url) { // 对于jar中文件包含file:前缀,需要去掉此类前缀,在此做标准化,since 3.0.8 解决中文或空格路径被编码的问题 return FileUtil.normalize(URLUtil.getDecodedPath(url)); } // 如果资源不存在,则返回一个拼接的资源绝对路径 final String classPath = ClassUtil.getClassPath(); if (null == classPath) { // throw new NullPointerException("ClassPath is null !"); // 在jar运行模式中,ClassPath有可能获取不到,此时返回原始相对路径(此时获取的文件为相对工作目录) return path; } // 资源不存在的情况下使用标准化路径有问题,使用原始路径拼接后标准化路径 return normalize(classPath.concat(Objects.requireNonNull(path))); } /** * 获取绝对路径,相对于ClassPath的目录
* 如果给定就是绝对路径,则返回原路径,原路径把所有\替换为/
* 兼容Spring风格的路径表示,例如:classpath:config/example.setting也会被识别后转换 * * @param path 相对路径 * @return 绝对路径 */ public static String getAbsolutePath(String path) { return getAbsolutePath(path, null); } /** * 获取标准的绝对路径 * * @param file 文件 * @return 绝对路径 */ public static String getAbsolutePath(File file) { if (file == null) { return null; } try { return file.getCanonicalPath(); } catch (IOException e) { return file.getAbsolutePath(); } } /** * 给定路径已经是绝对路径
* 此方法并没有针对路径做标准化,建议先执行{@link #normalize(String)}方法标准化路径后判断
* 绝对路径判断条件是: *
    *
  • 以/开头的路径
  • *
  • 满足类似于 c:/xxxxx,其中祖母随意,不区分大小写
  • *
  • 满足类似于 d:\xxxxx,其中祖母随意,不区分大小写
  • *
* * @param path 需要检查的Path * @return 是否已经是绝对路径 */ public static boolean isAbsolutePath(String path) { if (StrUtil.isEmpty(path)) { return false; } // 给定的路径已经是绝对路径了 return StrUtil.C_SLASH == path.charAt(0) || ReUtil.isMatch(PATTERN_PATH_ABSOLUTE, path); } /** * 判断是否为目录,如果path为null,则返回false * * @param path 文件路径 * @return 如果为目录true */ public static boolean isDirectory(String path) { return (null != path) && file(path).isDirectory(); } /** * 判断是否为目录,如果file为null,则返回false * * @param file 文件 * @return 如果为目录true */ public static boolean isDirectory(File file) { return (null != file) && file.isDirectory(); } /** * 判断是否为文件,如果path为null,则返回false * * @param path 文件路径 * @return 如果为文件true */ public static boolean isFile(String path) { return (null != path) && file(path).isFile(); } /** * 判断是否为文件,如果file为null,则返回false * * @param file 文件 * @return 如果为文件true */ public static boolean isFile(File file) { return (null != file) && file.isFile(); } /** * 检查两个文件是否是同一个文件
* 所谓文件相同,是指File对象是否指向同一个文件或文件夹 * * @param file1 文件1 * @param file2 文件2 * @return 是否相同 * @throws IORuntimeException IO异常 */ public static boolean equals(File file1, File file2) throws IORuntimeException { Assert.notNull(file1); Assert.notNull(file2); if (false == file1.exists() || false == file2.exists()) { // 两个文件都不存在判断其路径是否相同, 对于一个存在一个不存在的情况,一定不相同 return false == file1.exists()// && false == file2.exists()// && pathEquals(file1, file2); } return equals(file1.toPath(), file2.toPath()); } /** * 比较两个文件内容是否相同
* 首先比较长度,长度一致再比较内容
* 此方法来自Apache Commons io * * @param file1 文件1 * @param file2 文件2 * @return 两个文件内容一致返回true,否则false * @throws IORuntimeException IO异常 * @since 4.0.6 */ public static boolean contentEquals(File file1, File file2) throws IORuntimeException { boolean file1Exists = file1.exists(); if (file1Exists != file2.exists()) { return false; } if (false == file1Exists) { // 两个文件都不存在,返回true return true; } if (file1.isDirectory() || file2.isDirectory()) { // 不比较目录 throw new IORuntimeException("Can't compare directories, only files"); } if (file1.length() != file2.length()) { // 文件长度不同 return false; } if (equals(file1, file2)) { // 同一个文件 return true; } InputStream input1 = null; InputStream input2 = null; try { input1 = getInputStream(file1); input2 = getInputStream(file2); return IoUtil.contentEquals(input1, input2); } finally { IoUtil.close(input1); IoUtil.close(input2); } } // ----------------------------------------------------------------------- /** * 比较两个文件内容是否相同
* 首先比较长度,长度一致再比较内容,比较内容采用按行读取,每行比较
* 此方法来自Apache Commons io * * @param file1 文件1 * @param file2 文件2 * @param charset 编码,null表示使用平台默认编码 两个文件内容一致返回true,否则false * @return 是否相同 * @throws IORuntimeException IO异常 * @since 4.0.6 */ public static boolean contentEqualsIgnoreEOL(File file1, File file2, Charset charset) throws IORuntimeException { boolean file1Exists = file1.exists(); if (file1Exists != file2.exists()) { return false; } if (!file1Exists) { // 两个文件都不存在,返回true return true; } if (file1.isDirectory() || file2.isDirectory()) { // 不比较目录 throw new IORuntimeException("Can't compare directories, only files"); } if (equals(file1, file2)) { // 同一个文件 return true; } Reader input1 = null; Reader input2 = null; try { input1 = getReader(file1, charset); input2 = getReader(file2, charset); return IoUtil.contentEqualsIgnoreEOL(input1, input2); } finally { IoUtil.close(input1); IoUtil.close(input2); } } /** * 文件路径是否相同
* 取两个文件的绝对路径比较,在Windows下忽略大小写,在Linux下不忽略。 * * @param file1 文件1 * @param file2 文件2 * @return 文件路径是否相同 * @since 3.0.9 */ public static boolean pathEquals(File file1, File file2) { if (isWindows()) { // Windows环境 try { if (StrUtil.equalsIgnoreCase(file1.getCanonicalPath(), file2.getCanonicalPath())) { return true; } } catch (Exception e) { if (StrUtil.equalsIgnoreCase(file1.getAbsolutePath(), file2.getAbsolutePath())) { return true; } } } else { // 类Unix环境 try { if (StrUtil.equals(file1.getCanonicalPath(), file2.getCanonicalPath())) { return true; } } catch (Exception e) { if (StrUtil.equals(file1.getAbsolutePath(), file2.getAbsolutePath())) { return true; } } } return false; } /** * 获得最后一个文件路径分隔符的位置 * * @param filePath 文件路径 * @return 最后一个文件路径分隔符的位置 */ public static int lastIndexOfSeparator(String filePath) { if (StrUtil.isNotEmpty(filePath)) { int i = filePath.length(); char c; while (--i >= 0) { c = filePath.charAt(i); if (CharUtil.isFileSeparator(c)) { return i; } } } return -1; } /** * 判断文件是否被改动
* 如果文件对象为 null 或者文件不存在,被视为改动 * * @param file 文件对象 * @param lastModifyTime 上次的改动时间 * @return 是否被改动 * @deprecated 拼写错误,请使用{@link #isModified(File, long)} */ @Deprecated public static boolean isModifed(File file, long lastModifyTime) { return isModified(file, lastModifyTime); } /** * 判断文件是否被改动
* 如果文件对象为 null 或者文件不存在,被视为改动 * * @param file 文件对象 * @param lastModifyTime 上次的改动时间 * @return 是否被改动 */ public static boolean isModified(File file, long lastModifyTime) { if (null == file || false == file.exists()) { return true; } return file.lastModified() != lastModifyTime; } /** * 修复路径
* 如果原路径尾部有分隔符,则保留为标准分隔符(/),否则不保留 *
    *
  1. 1. 统一用 /
  2. *
  3. 2. 多个 / 转换为一个 /
  4. *
  5. 3. 去除左边空格
  6. *
  7. 4. .. 和 . 转换为绝对路径,当..多于已有路径时,直接返回根路径
  8. *
*

* 栗子: * *

	 * "/foo//" =》 "/foo/"
	 * "/foo/./" =》 "/foo/"
	 * "/foo/../bar" =》 "/bar"
	 * "/foo/../bar/" =》 "/bar/"
	 * "/foo/../bar/../baz" =》 "/baz"
	 * "/../" =》 "/"
	 * "foo/bar/.." =》 "foo"
	 * "foo/../bar" =》 "bar"
	 * "foo/../../bar" =》 "bar"
	 * "//server/foo/../bar" =》 "/server/bar"
	 * "//server/../bar" =》 "/bar"
	 * "C:\\foo\\..\\bar" =》 "C:/bar"
	 * "C:\\..\\bar" =》 "C:/bar"
	 * "~/foo/../bar/" =》 "~/bar/"
	 * "~/../bar" =》 普通用户运行是'bar的home目录',ROOT用户运行是'/bar'
	 * 
* * @param path 原路径 * @return 修复后的路径 */ public static String normalize(String path) { if (path == null) { return null; } //兼容Windows下的共享目录路径(原始路径如果以\\开头,则保留这种路径) if (path.startsWith("\\\\")) { return path; } // 兼容Spring风格的ClassPath路径,去除前缀,不区分大小写 String pathToUse = StrUtil.removePrefixIgnoreCase(path, URLUtil.CLASSPATH_URL_PREFIX); // 去除file:前缀 pathToUse = StrUtil.removePrefixIgnoreCase(pathToUse, URLUtil.FILE_URL_PREFIX); // 识别home目录形式,并转换为绝对路径 if (StrUtil.startWith(pathToUse, '~')) { pathToUse = getUserHomePath() + pathToUse.substring(1); } // 统一使用斜杠 pathToUse = pathToUse.replaceAll("[/\\\\]+", StrUtil.SLASH); // 去除开头空白符,末尾空白符合法,不去除 pathToUse = StrUtil.trimStart(pathToUse); String prefix = StrUtil.EMPTY; int prefixIndex = pathToUse.indexOf(StrUtil.COLON); if (prefixIndex > -1) { // 可能Windows风格路径 prefix = pathToUse.substring(0, prefixIndex + 1); if (StrUtil.startWith(prefix, StrUtil.C_SLASH)) { // 去除类似于/C:这类路径开头的斜杠 prefix = prefix.substring(1); } if (false == prefix.contains(StrUtil.SLASH)) { pathToUse = pathToUse.substring(prefixIndex + 1); } else { // 如果前缀中包含/,说明非Windows风格path prefix = StrUtil.EMPTY; } } if (pathToUse.startsWith(StrUtil.SLASH)) { prefix += StrUtil.SLASH; pathToUse = pathToUse.substring(1); } List pathList = StrUtil.split(pathToUse, StrUtil.C_SLASH); List pathElements = new LinkedList<>(); int tops = 0; String element; for (int i = pathList.size() - 1; i >= 0; i--) { element = pathList.get(i); // 只处理非.的目录,即只处理非当前目录 if (false == StrUtil.DOT.equals(element)) { if (StrUtil.DOUBLE_DOT.equals(element)) { tops++; } else { if (tops > 0) { // 有上级目录标记时按照个数依次跳过 tops--; } else { // Normal path element found. pathElements.add(0, element); } } } } // issue#1703@Github if (tops > 0 && StrUtil.isEmpty(prefix)) { // 只有相对路径补充开头的..,绝对路径直接忽略之 while (tops-- > 0) { //遍历完节点发现还有上级标注(即开头有一个或多个..),补充之 // Normal path element found. pathElements.add(0, StrUtil.DOUBLE_DOT); } } return prefix + CollUtil.join(pathElements, StrUtil.SLASH); } /** * 获得相对子路径 *

* 栗子: * *

	 * dirPath: d:/aaa/bbb    filePath: d:/aaa/bbb/ccc     =》    ccc
	 * dirPath: d:/Aaa/bbb    filePath: d:/aaa/bbb/ccc.txt     =》    ccc.txt
	 * 
* * @param rootDir 绝对父路径 * @param file 文件 * @return 相对子路径 */ public static String subPath(String rootDir, File file) { try { return subPath(rootDir, file.getCanonicalPath()); } catch (IOException e) { throw new IORuntimeException(e); } } /** * 获得相对子路径,忽略大小写 *

* 栗子: * *

	 * dirPath: d:/aaa/bbb    filePath: d:/aaa/bbb/ccc     =》    ccc
	 * dirPath: d:/Aaa/bbb    filePath: d:/aaa/bbb/ccc.txt     =》    ccc.txt
	 * dirPath: d:/Aaa/bbb    filePath: d:/aaa/bbb/     =》    ""
	 * 
* * @param dirPath 父路径 * @param filePath 文件路径 * @return 相对子路径 */ public static String subPath(String dirPath, String filePath) { if (StrUtil.isNotEmpty(dirPath) && StrUtil.isNotEmpty(filePath)) { dirPath = StrUtil.removeSuffix(normalize(dirPath), "/"); filePath = normalize(filePath); final String result = StrUtil.removePrefixIgnoreCase(filePath, dirPath); return StrUtil.removePrefix(result, "/"); } return filePath; } // -------------------------------------------------------------------------------------------- name start /** * 返回文件名 * * @param file 文件 * @return 文件名 * @see FileNameUtil#getName(File) * @since 4.1.13 */ public static String getName(File file) { return FileNameUtil.getName(file); } /** * 返回文件名
*
	 * "d:/test/aaa" 返回 "aaa"
	 * "/test/aaa.jpg" 返回 "aaa.jpg"
	 * 
* * @param filePath 文件 * @return 文件名 * @see FileNameUtil#getName(String) * @since 4.1.13 */ public static String getName(String filePath) { return FileNameUtil.getName(filePath); } /** * 获取文件后缀名,扩展名不带“.” * * @param file 文件 * @return 扩展名 * @see FileNameUtil#getSuffix(File) * @since 5.3.8 */ public static String getSuffix(File file) { return FileNameUtil.getSuffix(file); } /** * 获得文件后缀名,扩展名不带“.” * * @param fileName 文件名 * @return 扩展名 * @see FileNameUtil#getSuffix(String) * @since 5.3.8 */ public static String getSuffix(String fileName) { return FileNameUtil.getSuffix(fileName); } /** * 返回主文件名 * * @param file 文件 * @return 主文件名 * @see FileNameUtil#getPrefix(File) * @since 5.3.8 */ public static String getPrefix(File file) { return FileNameUtil.getPrefix(file); } /** * 返回主文件名 * * @param fileName 完整文件名 * @return 主文件名 * @see FileNameUtil#getPrefix(String) * @since 5.3.8 */ public static String getPrefix(String fileName) { return FileNameUtil.getPrefix(fileName); } /** * 返回主文件名 * * @param file 文件 * @return 主文件名 * @see FileNameUtil#mainName(File) */ public static String mainName(File file) { return FileNameUtil.mainName(file); } /** * 返回主文件名 * * @param fileName 完整文件名 * @return 主文件名 * @see FileNameUtil#mainName(String) */ public static String mainName(String fileName) { return FileNameUtil.mainName(fileName); } /** * 获取文件扩展名(后缀名),扩展名不带“.” * * @param file 文件 * @return 扩展名 * @see FileNameUtil#extName(File) */ public static String extName(File file) { return FileNameUtil.extName(file); } /** * 获得文件的扩展名(后缀名),扩展名不带“.” * * @param fileName 文件名 * @return 扩展名 * @see FileNameUtil#extName(String) */ public static String extName(String fileName) { return FileNameUtil.extName(fileName); } // -------------------------------------------------------------------------------------------- name end /** * 判断文件路径是否有指定后缀,忽略大小写
* 常用语判断扩展名 * * @param file 文件或目录 * @param suffix 后缀 * @return 是否有指定后缀 */ public static boolean pathEndsWith(File file, String suffix) { return file.getPath().toLowerCase().endsWith(suffix); } /** * 根据文件流的头部信息获得文件类型 * *
	 *      1、无法识别类型默认按照扩展名识别
	 *      2、xls、doc、msi头信息无法区分,按照扩展名区分
	 *      3、zip可能为docx、xlsx、pptx、jar、war头信息无法区分,按照扩展名区分
	 * 
* * @param file 文件 {@link File} * @return 类型,文件的扩展名,未找到为{@code null} * @throws IORuntimeException IO异常 * @see FileTypeUtil#getType(File) */ public static String getType(File file) throws IORuntimeException { return FileTypeUtil.getType(file); } // -------------------------------------------------------------------------------------------- in start /** * 获得输入流 * * @param file 文件 * @return 输入流 * @throws IORuntimeException 文件未找到 */ public static BufferedInputStream getInputStream(File file) throws IORuntimeException { return IoUtil.toBuffered(IoUtil.toStream(file)); } /** * 获得输入流 * * @param path 文件路径 * @return 输入流 * @throws IORuntimeException 文件未找到 */ public static BufferedInputStream getInputStream(String path) throws IORuntimeException { return getInputStream(file(path)); } /** * 获得BOM输入流,用于处理带BOM头的文件 * * @param file 文件 * @return 输入流 * @throws IORuntimeException 文件未找到 */ public static BOMInputStream getBOMInputStream(File file) throws IORuntimeException { try { return new BOMInputStream(Files.newInputStream(file.toPath())); } catch (IOException e) { throw new IORuntimeException(e); } } /** * 读取带BOM头的文件为Reader * * @param file 文件 * @return BufferedReader对象 * @since 5.5.8 */ public static BufferedReader getBOMReader(File file) { return IoUtil.getReader(getBOMInputStream(file)); } /** * 获得一个文件读取器 * * @param file 文件 * @return BufferedReader对象 * @throws IORuntimeException IO异常 */ public static BufferedReader getUtf8Reader(File file) throws IORuntimeException { return getReader(file, CharsetUtil.CHARSET_UTF_8); } /** * 获得一个文件读取器 * * @param path 文件路径 * @return BufferedReader对象 * @throws IORuntimeException IO异常 */ public static BufferedReader getUtf8Reader(String path) throws IORuntimeException { return getReader(path, CharsetUtil.CHARSET_UTF_8); } /** * 获得一个文件读取器 * * @param file 文件 * @param charsetName 字符集 * @return BufferedReader对象 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #getReader(File, Charset)} */ @Deprecated public static BufferedReader getReader(File file, String charsetName) throws IORuntimeException { return IoUtil.getReader(getInputStream(file), CharsetUtil.charset(charsetName)); } /** * 获得一个文件读取器 * * @param file 文件 * @param charset 字符集 * @return BufferedReader对象 * @throws IORuntimeException IO异常 */ public static BufferedReader getReader(File file, Charset charset) throws IORuntimeException { return IoUtil.getReader(getInputStream(file), charset); } /** * 获得一个文件读取器 * * @param path 绝对路径 * @param charsetName 字符集 * @return BufferedReader对象 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #getReader(String, Charset)} */ @Deprecated public static BufferedReader getReader(String path, String charsetName) throws IORuntimeException { return getReader(path, CharsetUtil.charset(charsetName)); } /** * 获得一个文件读取器 * * @param path 绝对路径 * @param charset 字符集 * @return BufferedReader对象 * @throws IORuntimeException IO异常 */ public static BufferedReader getReader(String path, Charset charset) throws IORuntimeException { return getReader(file(path), charset); } // -------------------------------------------------------------------------------------------- in end /** * 读取文件所有数据
* 文件的长度不能超过Integer.MAX_VALUE * * @param file 文件 * @return 字节码 * @throws IORuntimeException IO异常 */ public static byte[] readBytes(File file) throws IORuntimeException { return FileReader.create(file).readBytes(); } /** * 读取文件所有数据
* 文件的长度不能超过Integer.MAX_VALUE * * @param filePath 文件路径 * @return 字节码 * @throws IORuntimeException IO异常 * @since 3.2.0 */ public static byte[] readBytes(String filePath) throws IORuntimeException { return readBytes(file(filePath)); } /** * 读取文件内容 * * @param file 文件 * @return 内容 * @throws IORuntimeException IO异常 */ public static String readUtf8String(File file) throws IORuntimeException { return readString(file, CharsetUtil.CHARSET_UTF_8); } /** * 读取文件内容 * * @param path 文件路径 * @return 内容 * @throws IORuntimeException IO异常 */ public static String readUtf8String(String path) throws IORuntimeException { return readString(path, CharsetUtil.CHARSET_UTF_8); } /** * 读取文件内容 * * @param file 文件 * @param charsetName 字符集 * @return 内容 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #readString(File, Charset)} */ @Deprecated public static String readString(File file, String charsetName) throws IORuntimeException { return readString(file, CharsetUtil.charset(charsetName)); } /** * 读取文件内容 * * @param file 文件 * @param charset 字符集 * @return 内容 * @throws IORuntimeException IO异常 */ public static String readString(File file, Charset charset) throws IORuntimeException { return FileReader.create(file, charset).readString(); } /** * 读取文件内容 * * @param path 文件路径 * @param charsetName 字符集 * @return 内容 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #readString(String, Charset)} */ @Deprecated public static String readString(String path, String charsetName) throws IORuntimeException { return readString(path, CharsetUtil.charset(charsetName)); } /** * 读取文件内容 * * @param path 文件路径 * @param charset 字符集 * @return 内容 * @throws IORuntimeException IO异常 */ public static String readString(String path, Charset charset) throws IORuntimeException { return readString(file(path), charset); } /** * 读取文件内容 * * @param url 文件URL * @param charsetName 字符集 * @return 内容 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #readString(URL, Charset)} */ @Deprecated public static String readString(URL url, String charsetName) throws IORuntimeException { return readString(url, CharsetUtil.charset(charsetName)); } /** * 读取文件内容 * * @param url 文件URL * @param charset 字符集 * @return 内容 * @throws IORuntimeException IO异常 * @since 5.7.10 */ public static String readString(URL url, Charset charset) throws IORuntimeException { if (url == null) { throw new NullPointerException("Empty url provided!"); } InputStream in = null; try { in = url.openStream(); return IoUtil.read(in, charset); } catch (IOException e) { throw new IORuntimeException(e); } finally { IoUtil.close(in); } } /** * 从文件中读取每一行的UTF-8编码数据 * * @param 集合类型 * @param path 文件路径 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static > T readUtf8Lines(String path, T collection) throws IORuntimeException { return readLines(path, CharsetUtil.CHARSET_UTF_8, collection); } /** * 从文件中读取每一行数据 * * @param 集合类型 * @param path 文件路径 * @param charset 字符集 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 */ public static > T readLines(String path, String charset, T collection) throws IORuntimeException { return readLines(file(path), charset, collection); } /** * 从文件中读取每一行数据 * * @param 集合类型 * @param path 文件路径 * @param charset 字符集 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 */ public static > T readLines(String path, Charset charset, T collection) throws IORuntimeException { return readLines(file(path), charset, collection); } /** * 从文件中读取每一行数据,数据编码为UTF-8 * * @param 集合类型 * @param file 文件路径 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static > T readUtf8Lines(File file, T collection) throws IORuntimeException { return readLines(file, CharsetUtil.CHARSET_UTF_8, collection); } /** * 从文件中读取每一行数据 * * @param 集合类型 * @param file 文件路径 * @param charset 字符集 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 */ public static > T readLines(File file, String charset, T collection) throws IORuntimeException { return FileReader.create(file, CharsetUtil.charset(charset)).readLines(collection); } /** * 从文件中读取每一行数据 * * @param 集合类型 * @param file 文件路径 * @param charset 字符集 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 */ public static > T readLines(File file, Charset charset, T collection) throws IORuntimeException { return FileReader.create(file, charset).readLines(collection); } /** * 从文件中读取每一行数据,编码为UTF-8 * * @param 集合类型 * @param url 文件的URL * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 */ public static > T readUtf8Lines(URL url, T collection) throws IORuntimeException { return readLines(url, CharsetUtil.CHARSET_UTF_8, collection); } /** * 从文件中读取每一行数据 * * @param 集合类型 * @param url 文件的URL * @param charsetName 字符集 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #readLines(URL, Charset, Collection)} */ @Deprecated public static > T readLines(URL url, String charsetName, T collection) throws IORuntimeException { return readLines(url, CharsetUtil.charset(charsetName), collection); } /** * 从文件中读取每一行数据 * * @param 集合类型 * @param url 文件的URL * @param charset 字符集 * @param collection 集合 * @return 文件中的每行内容的集合 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static > T readLines(URL url, Charset charset, T collection) throws IORuntimeException { InputStream in = null; try { in = url.openStream(); return IoUtil.readLines(in, charset, collection); } catch (IOException e) { throw new IORuntimeException(e); } finally { IoUtil.close(in); } } /** * 从文件中读取每一行数据 * * @param url 文件的URL * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 */ public static List readUtf8Lines(URL url) throws IORuntimeException { return readLines(url, CharsetUtil.CHARSET_UTF_8); } /** * 从文件中读取每一行数据 * * @param url 文件的URL * @param charsetName 字符集 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #readLines(URL, Charset)} */ @Deprecated public static List readLines(URL url, String charsetName) throws IORuntimeException { return readLines(url, CharsetUtil.charset(charsetName)); } /** * 从文件中读取每一行数据 * * @param url 文件的URL * @param charset 字符集 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 */ public static List readLines(URL url, Charset charset) throws IORuntimeException { return readLines(url, charset, new ArrayList<>()); } /** * 从文件中读取每一行数据,编码为UTF-8 * * @param path 文件路径 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static List readUtf8Lines(String path) throws IORuntimeException { return readLines(path, CharsetUtil.CHARSET_UTF_8); } /** * 从文件中读取每一行数据 * * @param path 文件路径 * @param charset 字符集 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 */ public static List readLines(String path, String charset) throws IORuntimeException { return readLines(path, charset, new ArrayList<>()); } /** * 从文件中读取每一行数据 * * @param path 文件路径 * @param charset 字符集 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static List readLines(String path, Charset charset) throws IORuntimeException { return readLines(path, charset, new ArrayList<>()); } /** * 从文件中读取每一行数据 * * @param file 文件 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static List readUtf8Lines(File file) throws IORuntimeException { return readLines(file, CharsetUtil.CHARSET_UTF_8); } /** * 从文件中读取每一行数据 * * @param file 文件 * @param filter 过滤器 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static List readUtf8Lines(File file, Predicate filter) throws IORuntimeException { return readLines(file, CharsetUtil.CHARSET_UTF_8, filter); } /** * 从文件中读取每一行数据 * * @param file 文件 * @param charset 字符集 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 */ public static List readLines(File file, String charset) throws IORuntimeException { return readLines(file, charset, new ArrayList<>()); } /** * 从文件中读取每一行数据 * * @param file 文件 * @param charset 字符集 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 */ public static List readLines(File file, Charset charset) throws IORuntimeException { return readLines(file, charset, new ArrayList<>()); } /** * 从文件中读取每一行数据 * * @param file 文件 * @param charset 字符集 * @param filter 过滤器 * @return 文件中的每行内容的集合List * @throws IORuntimeException IO异常 */ public static List readLines(File file, Charset charset, Predicate filter) throws IORuntimeException { final List result = new ArrayList<>(); readLines(file, charset, (LineHandler) line -> { if (filter.test(line)) { result.add(line); } }); return result; } /** * 按行处理文件内容,编码为UTF-8 * * @param file 文件 * @param lineHandler {@link LineHandler}行处理器 * @throws IORuntimeException IO异常 */ public static void readUtf8Lines(File file, LineHandler lineHandler) throws IORuntimeException { readLines(file, CharsetUtil.CHARSET_UTF_8, lineHandler); } /** * 按行处理文件内容 * * @param file 文件 * @param charset 编码 * @param lineHandler {@link LineHandler}行处理器 * @throws IORuntimeException IO异常 */ public static void readLines(File file, Charset charset, LineHandler lineHandler) throws IORuntimeException { FileReader.create(file, charset).readLines(lineHandler); } /** * 按行处理文件内容 * * @param file {@link RandomAccessFile}文件 * @param charset 编码 * @param lineHandler {@link LineHandler}行处理器 * @throws IORuntimeException IO异常 * @since 4.5.2 */ public static void readLines(RandomAccessFile file, Charset charset, LineHandler lineHandler) { String line; try { while ((line = file.readLine()) != null) { lineHandler.handle(CharsetUtil.convert(line, CharsetUtil.CHARSET_ISO_8859_1, charset)); } } catch (IOException e) { throw new IORuntimeException(e); } } /** * 单行处理文件内容 * * @param file {@link RandomAccessFile}文件 * @param charset 编码 * @param lineHandler {@link LineHandler}行处理器 * @throws IORuntimeException IO异常 * @since 4.5.2 */ public static void readLine(RandomAccessFile file, Charset charset, LineHandler lineHandler) { final String line = readLine(file, charset); if (null != line) { lineHandler.handle(line); } } /** * 单行处理文件内容 * * @param file {@link RandomAccessFile}文件 * @param charset 编码 * @return 行内容 * @throws IORuntimeException IO异常 * @since 4.5.18 */ public static String readLine(RandomAccessFile file, Charset charset) { String line; try { line = file.readLine(); } catch (IOException e) { throw new IORuntimeException(e); } if (null != line) { return CharsetUtil.convert(line, CharsetUtil.CHARSET_ISO_8859_1, charset); } return null; } /** * 按照给定的readerHandler读取文件中的数据 * * @param 集合类型 * @param readerHandler Reader处理类 * @param path 文件的绝对路径 * @return 从文件中load出的数据 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static T loadUtf8(String path, ReaderHandler readerHandler) throws IORuntimeException { return load(path, CharsetUtil.CHARSET_UTF_8, readerHandler); } /** * 按照给定的readerHandler读取文件中的数据 * * @param 集合类型 * @param readerHandler Reader处理类 * @param path 文件的绝对路径 * @param charset 字符集 * @return 从文件中load出的数据 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static T load(String path, String charset, ReaderHandler readerHandler) throws IORuntimeException { return FileReader.create(file(path), CharsetUtil.charset(charset)).read(readerHandler); } /** * 按照给定的readerHandler读取文件中的数据 * * @param 集合类型 * @param readerHandler Reader处理类 * @param path 文件的绝对路径 * @param charset 字符集 * @return 从文件中load出的数据 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static T load(String path, Charset charset, ReaderHandler readerHandler) throws IORuntimeException { return FileReader.create(file(path), charset).read(readerHandler); } /** * 按照给定的readerHandler读取文件中的数据 * * @param 集合类型 * @param readerHandler Reader处理类 * @param file 文件 * @return 从文件中load出的数据 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static T loadUtf8(File file, ReaderHandler readerHandler) throws IORuntimeException { return load(file, CharsetUtil.CHARSET_UTF_8, readerHandler); } /** * 按照给定的readerHandler读取文件中的数据 * * @param 集合类型 * @param readerHandler Reader处理类 * @param file 文件 * @param charset 字符集 * @return 从文件中load出的数据 * @throws IORuntimeException IO异常 * @since 3.1.1 */ public static T load(File file, Charset charset, ReaderHandler readerHandler) throws IORuntimeException { return FileReader.create(file, charset).read(readerHandler); } // -------------------------------------------------------------------------------------------- out start /** * 获得一个输出流对象 * * @param file 文件 * @return 输出流对象 * @throws IORuntimeException IO异常 */ public static BufferedOutputStream getOutputStream(File file) throws IORuntimeException { final OutputStream out; try { out = Files.newOutputStream(touch(file).toPath()); } catch (final IOException e) { throw new IORuntimeException(e); } return IoUtil.toBuffered(out); } /** * 获得一个输出流对象 * * @param path 输出到的文件路径,绝对路径 * @return 输出流对象 * @throws IORuntimeException IO异常 */ public static BufferedOutputStream getOutputStream(String path) throws IORuntimeException { return getOutputStream(touch(path)); } /** * 获得一个带缓存的写入对象 * * @param path 输出路径,绝对路径 * @param charsetName 字符集 * @param isAppend 是否追加 * @return BufferedReader对象 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #getWriter(String, Charset, boolean)} */ @Deprecated public static BufferedWriter getWriter(String path, String charsetName, boolean isAppend) throws IORuntimeException { return getWriter(path, Charset.forName(charsetName), isAppend); } /** * 获得一个带缓存的写入对象 * * @param path 输出路径,绝对路径 * @param charset 字符集 * @param isAppend 是否追加 * @return BufferedReader对象 * @throws IORuntimeException IO异常 */ public static BufferedWriter getWriter(String path, Charset charset, boolean isAppend) throws IORuntimeException { return getWriter(touch(path), charset, isAppend); } /** * 获得一个带缓存的写入对象 * * @param file 输出文件 * @param charsetName 字符集 * @param isAppend 是否追加 * @return BufferedReader对象 * @throws IORuntimeException IO异常 * @deprecated 请使用 {@link #getWriter(File, Charset, boolean)} */ @Deprecated public static BufferedWriter getWriter(File file, String charsetName, boolean isAppend) throws IORuntimeException { return getWriter(file, Charset.forName(charsetName), isAppend); } /** * 获得一个带缓存的写入对象 * * @param file 输出文件 * @param charset 字符集 * @param isAppend 是否追加 * @return BufferedReader对象 * @throws IORuntimeException IO异常 */ public static BufferedWriter getWriter(File file, Charset charset, boolean isAppend) throws IORuntimeException { return FileWriter.create(file, charset).getWriter(isAppend); } /** * 获得一个打印写入对象,可以有print * * @param path 输出路径,绝对路径 * @param charset 字符集 * @param isAppend 是否追加 * @return 打印对象 * @throws IORuntimeException IO异常 */ public static PrintWriter getPrintWriter(String path, String charset, boolean isAppend) throws IORuntimeException { return new PrintWriter(getWriter(path, charset, isAppend)); } /** * 获得一个打印写入对象,可以有print * * @param path 输出路径,绝对路径 * @param charset 字符集 * @param isAppend 是否追加 * @return 打印对象 * @throws IORuntimeException IO异常 * @since 4.1.1 */ public static PrintWriter getPrintWriter(String path, Charset charset, boolean isAppend) throws IORuntimeException { return new PrintWriter(getWriter(path, charset, isAppend)); } /** * 获得一个打印写入对象,可以有print * * @param file 文件 * @param charset 字符集 * @param isAppend 是否追加 * @return 打印对象 * @throws IORuntimeException IO异常 */ public static PrintWriter getPrintWriter(File file, String charset, boolean isAppend) throws IORuntimeException { return new PrintWriter(getWriter(file, charset, isAppend)); } /** * 获得一个打印写入对象,可以有print * * @param file 文件 * @param charset 字符集 * @param isAppend 是否追加 * @return 打印对象 * @throws IORuntimeException IO异常 * @since 5.4.3 */ public static PrintWriter getPrintWriter(File file, Charset charset, boolean isAppend) throws IORuntimeException { return new PrintWriter(getWriter(file, charset, isAppend)); } /** * 获取当前系统的换行分隔符 * *
	 * Windows: \r\n
	 * Mac: \r
	 * Linux: \n
	 * 
* * @return 换行符 * @since 4.0.5 */ public static String getLineSeparator() { return System.lineSeparator(); // return System.getProperty("line.separator"); } // -------------------------------------------------------------------------------------------- out end /** * 将String写入文件,覆盖模式,字符集为UTF-8 * * @param content 写入的内容 * @param path 文件路径 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File writeUtf8String(String content, String path) throws IORuntimeException { return writeString(content, path, CharsetUtil.CHARSET_UTF_8); } /** * 将String写入文件,覆盖模式,字符集为UTF-8 * * @param content 写入的内容 * @param file 文件 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File writeUtf8String(String content, File file) throws IORuntimeException { return writeString(content, file, CharsetUtil.CHARSET_UTF_8); } /** * 将String写入文件,覆盖模式 * * @param content 写入的内容 * @param path 文件路径 * @param charset 字符集 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File writeString(String content, String path, String charset) throws IORuntimeException { return writeString(content, touch(path), charset); } /** * 将String写入文件,覆盖模式 * * @param content 写入的内容 * @param path 文件路径 * @param charset 字符集 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File writeString(String content, String path, Charset charset) throws IORuntimeException { return writeString(content, touch(path), charset); } /** * 将String写入文件,覆盖模式 * * @param content 写入的内容 * @param file 文件 * @param charset 字符集 * @return 被写入的文件 * @throws IORuntimeException IO异常 */ public static File writeString(String content, File file, String charset) throws IORuntimeException { return FileWriter.create(file, CharsetUtil.charset(charset)).write(content); } /** * 将String写入文件,覆盖模式 * * @param content 写入的内容 * @param file 文件 * @param charset 字符集 * @return 被写入的文件 * @throws IORuntimeException IO异常 */ public static File writeString(String content, File file, Charset charset) throws IORuntimeException { return FileWriter.create(file, charset).write(content); } /** * 将String写入文件,UTF-8编码追加模式 * * @param content 写入的内容 * @param path 文件路径 * @return 写入的文件 * @throws IORuntimeException IO异常 * @since 3.1.2 */ public static File appendUtf8String(String content, String path) throws IORuntimeException { return appendString(content, path, CharsetUtil.CHARSET_UTF_8); } /** * 将String写入文件,追加模式 * * @param content 写入的内容 * @param path 文件路径 * @param charset 字符集 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File appendString(String content, String path, String charset) throws IORuntimeException { return appendString(content, touch(path), charset); } /** * 将String写入文件,追加模式 * * @param content 写入的内容 * @param path 文件路径 * @param charset 字符集 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File appendString(String content, String path, Charset charset) throws IORuntimeException { return appendString(content, touch(path), charset); } /** * 将String写入文件,UTF-8编码追加模式 * * @param content 写入的内容 * @param file 文件 * @return 写入的文件 * @throws IORuntimeException IO异常 * @since 3.1.2 */ public static File appendUtf8String(String content, File file) throws IORuntimeException { return appendString(content, file, CharsetUtil.CHARSET_UTF_8); } /** * 将String写入文件,追加模式 * * @param content 写入的内容 * @param file 文件 * @param charset 字符集 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File appendString(String content, File file, String charset) throws IORuntimeException { return FileWriter.create(file, CharsetUtil.charset(charset)).append(content); } /** * 将String写入文件,追加模式 * * @param content 写入的内容 * @param file 文件 * @param charset 字符集 * @return 写入的文件 * @throws IORuntimeException IO异常 */ public static File appendString(String content, File file, Charset charset) throws IORuntimeException { return FileWriter.create(file, charset).append(content); } /** * 将列表写入文件,覆盖模式,编码为UTF-8 * * @param 集合元素类型 * @param list 列表 * @param path 绝对路径 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 3.2.0 */ public static File writeUtf8Lines(Collection list, String path) throws IORuntimeException { return writeLines(list, path, CharsetUtil.CHARSET_UTF_8); } /** * 将列表写入文件,覆盖模式,编码为UTF-8 * * @param 集合元素类型 * @param list 列表 * @param file 绝对路径 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 3.2.0 */ public static File writeUtf8Lines(Collection list, File file) throws IORuntimeException { return writeLines(list, file, CharsetUtil.CHARSET_UTF_8); } /** * 将列表写入文件,覆盖模式 * * @param 集合元素类型 * @param list 列表 * @param path 绝对路径 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeLines(Collection list, String path, String charset) throws IORuntimeException { return writeLines(list, path, charset, false); } /** * 将列表写入文件,覆盖模式 * * @param 集合元素类型 * @param list 列表 * @param path 绝对路径 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeLines(Collection list, String path, Charset charset) throws IORuntimeException { return writeLines(list, path, charset, false); } /** * 将列表写入文件,覆盖模式 * * @param 集合元素类型 * @param list 列表 * @param file 文件 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 4.2.0 */ public static File writeLines(Collection list, File file, String charset) throws IORuntimeException { return writeLines(list, file, charset, false); } /** * 将列表写入文件,覆盖模式 * * @param 集合元素类型 * @param list 列表 * @param file 文件 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 4.2.0 */ public static File writeLines(Collection list, File file, Charset charset) throws IORuntimeException { return writeLines(list, file, charset, false); } /** * 将列表写入文件,追加模式 * * @param 集合元素类型 * @param list 列表 * @param file 文件 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 3.1.2 */ public static File appendUtf8Lines(Collection list, File file) throws IORuntimeException { return appendLines(list, file, CharsetUtil.CHARSET_UTF_8); } /** * 将列表写入文件,追加模式 * * @param 集合元素类型 * @param list 列表 * @param path 文件路径 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 3.1.2 */ public static File appendUtf8Lines(Collection list, String path) throws IORuntimeException { return appendLines(list, path, CharsetUtil.CHARSET_UTF_8); } /** * 将列表写入文件,追加模式 * * @param 集合元素类型 * @param list 列表 * @param path 绝对路径 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File appendLines(Collection list, String path, String charset) throws IORuntimeException { return writeLines(list, path, charset, true); } /** * 将列表写入文件,追加模式 * * @param 集合元素类型 * @param list 列表 * @param file 文件 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 3.1.2 */ public static File appendLines(Collection list, File file, String charset) throws IORuntimeException { return writeLines(list, file, charset, true); } /** * 将列表写入文件,追加模式 * * @param 集合元素类型 * @param list 列表 * @param path 绝对路径 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File appendLines(Collection list, String path, Charset charset) throws IORuntimeException { return writeLines(list, path, charset, true); } /** * 将列表写入文件,追加模式,策略为: *
    *
  • 当文件为空,从开头追加,尾部不加空行
  • *
  • 当有内容,换行追加,尾部不加空行
  • *
  • 当有内容,并末尾有空行,依旧换行追加
  • *
* * @param 集合元素类型 * @param list 列表 * @param file 文件 * @param charset 字符集 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 3.1.2 */ public static File appendLines(Collection list, File file, Charset charset) throws IORuntimeException { return writeLines(list, file, charset, true); } /** * 将列表写入文件 * * @param 集合元素类型 * @param list 列表 * @param path 文件路径 * @param charset 字符集 * @param isAppend 是否追加 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeLines(Collection list, String path, String charset, boolean isAppend) throws IORuntimeException { return writeLines(list, file(path), charset, isAppend); } /** * 将列表写入文件 * * @param 集合元素类型 * @param list 列表 * @param path 文件路径 * @param charset 字符集 * @param isAppend 是否追加 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeLines(Collection list, String path, Charset charset, boolean isAppend) throws IORuntimeException { return writeLines(list, file(path), charset, isAppend); } /** * 将列表写入文件 * * @param 集合元素类型 * @param list 列表 * @param file 文件 * @param charset 字符集 * @param isAppend 是否追加 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeLines(Collection list, File file, String charset, boolean isAppend) throws IORuntimeException { return FileWriter.create(file, CharsetUtil.charset(charset)).writeLines(list, isAppend); } /** * 将列表写入文件 * * @param 集合元素类型 * @param list 列表 * @param file 文件 * @param charset 字符集 * @param isAppend 是否追加 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeLines(Collection list, File file, Charset charset, boolean isAppend) throws IORuntimeException { return FileWriter.create(file, charset).writeLines(list, isAppend); } /** * 将Map写入文件,每个键值对为一行,一行中键与值之间使用kvSeparator分隔 * * @param map Map * @param file 文件 * @param kvSeparator 键和值之间的分隔符,如果传入null使用默认分隔符" = " * @param isAppend 是否追加 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 4.0.5 */ public static File writeUtf8Map(Map map, File file, String kvSeparator, boolean isAppend) throws IORuntimeException { return FileWriter.create(file, CharsetUtil.CHARSET_UTF_8).writeMap(map, kvSeparator, isAppend); } /** * 将Map写入文件,每个键值对为一行,一行中键与值之间使用kvSeparator分隔 * * @param map Map * @param file 文件 * @param charset 字符集编码 * @param kvSeparator 键和值之间的分隔符,如果传入null使用默认分隔符" = " * @param isAppend 是否追加 * @return 目标文件 * @throws IORuntimeException IO异常 * @since 4.0.5 */ public static File writeMap(Map map, File file, Charset charset, String kvSeparator, boolean isAppend) throws IORuntimeException { return FileWriter.create(file, charset).writeMap(map, kvSeparator, isAppend); } /** * 写数据到文件中
* 文件路径如果是相对路径,则相对ClassPath * * @param data 数据 * @param path 相对ClassPath的目录或者绝对路径目录 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeBytes(byte[] data, String path) throws IORuntimeException { return writeBytes(data, touch(path)); } /** * 写数据到文件中 * * @param dest 目标文件 * @param data 数据 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeBytes(byte[] data, File dest) throws IORuntimeException { return writeBytes(data, dest, 0, data.length, false); } /** * 写入数据到文件 * * @param data 数据 * @param dest 目标文件 * @param off 数据开始位置 * @param len 数据长度 * @param isAppend 是否追加模式 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeBytes(byte[] data, File dest, int off, int len, boolean isAppend) throws IORuntimeException { return FileWriter.create(dest).write(data, off, len, isAppend); } /** * 将流的内容写入文件
* 此方法会自动关闭输入流 * * @param dest 目标文件 * @param in 输入流 * @return dest * @throws IORuntimeException IO异常 */ public static File writeFromStream(InputStream in, File dest) throws IORuntimeException { return writeFromStream(in, dest, true); } /** * 将流的内容写入文件 * * @param dest 目标文件 * @param in 输入流 * @param isCloseIn 是否关闭输入流 * @return dest * @throws IORuntimeException IO异常 * @since 5.5.6 */ public static File writeFromStream(InputStream in, File dest, boolean isCloseIn) throws IORuntimeException { return FileWriter.create(dest).writeFromStream(in, isCloseIn); } /** * 将流的内容写入文件
* 此方法会自动关闭输入流 * * @param in 输入流 * @param fullFilePath 文件绝对路径 * @return 目标文件 * @throws IORuntimeException IO异常 */ public static File writeFromStream(InputStream in, String fullFilePath) throws IORuntimeException { return writeFromStream(in, touch(fullFilePath)); } /** * 将文件写入流中,此方法不会关闭输出流 * * @param file 文件 * @param out 流 * @return 写出的流byte数 * @throws IORuntimeException IO异常 */ public static long writeToStream(File file, OutputStream out) throws IORuntimeException { return FileReader.create(file).writeToStream(out); } /** * 将路径对应文件写入流中,此方法不会关闭输出流 * * @param fullFilePath 文件绝对路径 * @param out 输出流 * @return 写出的流byte数 * @throws IORuntimeException IO异常 */ public static long writeToStream(String fullFilePath, OutputStream out) throws IORuntimeException { return writeToStream(touch(fullFilePath), out); } /** * 可读的文件大小 * * @param file 文件 * @return 大小 */ public static String readableFileSize(File file) { return readableFileSize(file.length()); } /** * 可读的文件大小
* 参考 http://stackoverflow.com/questions/3263892/format-file-size-as-mb-gb-etc * * @param size Long类型大小 * @return 大小 * @see DataSizeUtil#format(long) */ public static String readableFileSize(long size) { return DataSizeUtil.format(size); } /** * 转换文件编码
* 此方法用于转换文件编码,读取的文件实际编码必须与指定的srcCharset编码一致,否则导致乱码 * * @param file 文件 * @param srcCharset 原文件的编码,必须与文件内容的编码保持一致 * @param destCharset 转码后的编码 * @return 被转换编码的文件 * @see CharsetUtil#convert(File, Charset, Charset) * @since 3.1.0 */ public static File convertCharset(File file, Charset srcCharset, Charset destCharset) { return CharsetUtil.convert(file, srcCharset, destCharset); } /** * 转换换行符
* 将给定文件的换行符转换为指定换行符 * * @param file 文件 * @param charset 编码 * @param lineSeparator 换行符枚举{@link LineSeparator} * @return 被修改的文件 * @since 3.1.0 */ public static File convertLineSeparator(File file, Charset charset, LineSeparator lineSeparator) { final List lines = readLines(file, charset); return FileWriter.create(file, charset).writeLines(lines, lineSeparator, false); } /** * 清除文件名中的在Windows下不支持的非法字符,包括: \ / : * ? " < > | * * @param fileName 文件名(必须不包括路径,否则路径符将被替换) * @return 清理后的文件名 * @see FileNameUtil#cleanInvalid(String) * @since 3.3.1 */ public static String cleanInvalid(String fileName) { return FileNameUtil.cleanInvalid(fileName); } /** * 文件名中是否包含在Windows下不支持的非法字符,包括: \ / : * ? " < > | * * @param fileName 文件名(必须不包括路径,否则路径符将被替换) * @return 是否包含非法字符 * @see FileNameUtil#containsInvalid(String) * @since 3.3.1 */ public static boolean containsInvalid(String fileName) { return FileNameUtil.containsInvalid(fileName); } /** * 计算文件CRC32校验码 * * @param file 文件,不能为目录 * @return CRC32值 * @throws IORuntimeException IO异常 * @since 4.0.6 */ public static long checksumCRC32(File file) throws IORuntimeException { return checksum(file, new CRC32()).getValue(); } /** * 计算文件校验码 * * @param file 文件,不能为目录 * @param checksum {@link Checksum} * @return Checksum * @throws IORuntimeException IO异常 * @since 4.0.6 */ public static Checksum checksum(File file, Checksum checksum) throws IORuntimeException { Assert.notNull(file, "File is null !"); if (file.isDirectory()) { throw new IllegalArgumentException("Checksums can't be computed on directories"); } try { return IoUtil.checksum(Files.newInputStream(file.toPath()), checksum); } catch (IOException e) { throw new IORuntimeException(e); } } /** * 获取Web项目下的web root路径
* 原理是首先获取ClassPath路径,由于在web项目中ClassPath位于 WEB-INF/classes/下,故向上获取两级目录即可。 * * @return web root路径 * @since 4.0.13 */ public static File getWebRoot() { final String classPath = ClassUtil.getClassPath(); if (StrUtil.isNotBlank(classPath)) { return getParent(file(classPath), 2); } return null; } /** * 获取指定层级的父路径 * *
	 * getParent("d:/aaa/bbb/cc/ddd", 0) -》 "d:/aaa/bbb/cc/ddd"
	 * getParent("d:/aaa/bbb/cc/ddd", 2) -》 "d:/aaa/bbb"
	 * getParent("d:/aaa/bbb/cc/ddd", 4) -》 "d:/"
	 * getParent("d:/aaa/bbb/cc/ddd", 5) -》 null
	 * 
* * @param filePath 目录或文件路径 * @param level 层级 * @return 路径File,如果不存在返回null * @since 4.1.2 */ public static String getParent(String filePath, int level) { final File parent = getParent(file(filePath), level); try { return null == parent ? null : parent.getCanonicalPath(); } catch (IOException e) { throw new IORuntimeException(e); } } /** * 获取指定层级的父路径 * *
	 * getParent(file("d:/aaa/bbb/cc/ddd", 0)) -》 "d:/aaa/bbb/cc/ddd"
	 * getParent(file("d:/aaa/bbb/cc/ddd", 2)) -》 "d:/aaa/bbb"
	 * getParent(file("d:/aaa/bbb/cc/ddd", 4)) -》 "d:/"
	 * getParent(file("d:/aaa/bbb/cc/ddd", 5)) -》 null
	 * 
* * @param file 目录或文件 * @param level 层级 * @return 路径File,如果不存在返回null * @since 4.1.2 */ public static File getParent(File file, int level) { if (level < 1 || null == file) { return file; } File parentFile; try { parentFile = file.getCanonicalFile().getParentFile(); } catch (IOException e) { throw new IORuntimeException(e); } if (1 == level) { return parentFile; } return getParent(parentFile, level - 1); } /** * 检查父完整路径是否为自路径的前半部分,如果不是说明不是子路径,可能存在slip注入。 *

* 见http://blog.nsfocus.net/zip-slip-2/ * * @param parentFile 父文件或目录 * @param file 子文件或目录 * @return 子文件或目录 * @throws IllegalArgumentException 检查创建的子文件不在父目录中抛出此异常 */ public static File checkSlip(File parentFile, File file) throws IllegalArgumentException { if (null != parentFile && null != file) { if (false == isSub(parentFile, file)) { throw new IllegalArgumentException("New file is outside of the parent dir: " + file.getName()); } } return file; } /** * 根据文件扩展名获得MimeType * * @param filePath 文件路径或文件名 * @return MimeType * @since 4.1.15 */ public static String getMimeType(String filePath) { if (StrUtil.isBlank(filePath)) { return null; } // 补充一些常用的mimeType if (StrUtil.endWithIgnoreCase(filePath, ".css")) { return "text/css"; } else if (StrUtil.endWithIgnoreCase(filePath, ".js")) { return "application/x-javascript"; } else if (StrUtil.endWithIgnoreCase(filePath, ".rar")) { return "application/x-rar-compressed"; } else if (StrUtil.endWithIgnoreCase(filePath, ".7z")) { return "application/x-7z-compressed"; } else if (StrUtil.endWithIgnoreCase(filePath, ".wgt")) { return "application/widget"; } else if (StrUtil.endWithIgnoreCase(filePath, ".webp")) { // JDK8不支持 return "image/webp"; } String contentType = URLConnection.getFileNameMap().getContentTypeFor(filePath); if (null == contentType) { contentType = getMimeType(Paths.get(filePath)); } return contentType; } /** * 判断是否为符号链接文件 * * @param file 被检查的文件 * @return 是否为符号链接文件 * @since 4.4.2 */ public static boolean isSymlink(File file) { return isSymlink(file.toPath()); } /** * 判断给定的目录是否为给定文件或文件夹的子目录 * * @param parent 父目录 * @param sub 子目录 * @return 子目录是否为父目录的子目录 * @since 4.5.4 */ public static boolean isSub(File parent, File sub) { Assert.notNull(parent); Assert.notNull(sub); return isSub(parent.toPath(), sub.toPath()); } /** * 创建{@link RandomAccessFile} * * @param path 文件Path * @param mode 模式,见{@link FileMode} * @return {@link RandomAccessFile} * @since 4.5.2 */ public static RandomAccessFile createRandomAccessFile(Path path, FileMode mode) { return createRandomAccessFile(path.toFile(), mode); } /** * 创建{@link RandomAccessFile} * * @param file 文件 * @param mode 模式,见{@link FileMode} * @return {@link RandomAccessFile} * @since 4.5.2 */ public static RandomAccessFile createRandomAccessFile(File file, FileMode mode) { try { return new RandomAccessFile(file, mode.name()); } catch (FileNotFoundException e) { throw new IORuntimeException(e); } } /** * 文件内容跟随器,实现类似Linux下"tail -f"命令功能
* 此方法会阻塞当前线程 * * @param file 文件 * @param handler 行处理器 */ public static void tail(File file, LineHandler handler) { tail(file, CharsetUtil.CHARSET_UTF_8, handler); } /** * 文件内容跟随器,实现类似Linux下"tail -f"命令功能
* 此方法会阻塞当前线程 * * @param file 文件 * @param charset 编码 * @param handler 行处理器 */ public static void tail(File file, Charset charset, LineHandler handler) { new Tailer(file, charset, handler).start(); } /** * 文件内容跟随器,实现类似Linux下"tail -f"命令功能
* 此方法会阻塞当前线程 * * @param file 文件 * @param charset 编码 */ public static void tail(File file, Charset charset) { tail(file, charset, Tailer.CONSOLE_HANDLER); } /** * 根据压缩包中的路径构建目录结构,在Win下直接构建,在Linux下拆分路径单独构建 * * @param outFile 最外部路径 * @param fileName 文件名,可以包含路径 * @return 文件或目录 * @since 5.0.5 */ private static File buildFile(File outFile, String fileName) { // 替换Windows路径分隔符为Linux路径分隔符,便于统一处理 fileName = fileName.replace('\\', '/'); if (false == isWindows() // 检查文件名中是否包含"/",不考虑以"/"结尾的情况 && fileName.lastIndexOf(CharUtil.SLASH, fileName.length() - 2) > 0) { // 在Linux下多层目录创建存在问题,/会被当成文件名的一部分,此处做处理 // 使用/拆分路径(zip中无\),级联创建父目录 final List pathParts = StrUtil.split(fileName, '/', false, true); final int lastPartIndex = pathParts.size() - 1;//目录个数 for (int i = 0; i < lastPartIndex; i++) { //由于路径拆分,slip不检查,在最后一步检查 outFile = new File(outFile, pathParts.get(i)); } //noinspection ResultOfMethodCallIgnored outFile.mkdirs(); // 最后一个部分如果非空,作为文件名 fileName = pathParts.get(lastPartIndex); } return new File(outFile, fileName); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy