В моем приложении я хочу сохранить копию определенного файла с другим именем (которое я получаю от пользователя)
Мне действительно нужно открыть содержимое файла и записать его в другой файл?
Каков наилучший способ сделать это?
В моем приложении я хочу сохранить копию определенного файла с другим именем (которое я получаю от пользователя)
Мне действительно нужно открыть содержимое файла и записать его в другой файл?
Каков наилучший способ сделать это?
Ответы:
Чтобы скопировать файл и сохранить его по своему пути назначения, вы можете использовать метод ниже.
public static void copy(File src, File dst) throws IOException {
InputStream in = new FileInputStream(src);
try {
OutputStream out = new FileOutputStream(dst);
try {
// Transfer bytes from in to out
byte[] buf = new byte[1024];
int len;
while ((len = in.read(buf)) > 0) {
out.write(buf, 0, len);
}
} finally {
out.close();
}
} finally {
in.close();
}
}
В API 19+ вы можете использовать Java Automatic Resource Management:
public static void copy(File src, File dst) throws IOException {
try (InputStream in = new FileInputStream(src)) {
try (OutputStream out = new FileOutputStream(dst)) {
// Transfer bytes from in to out
byte[] buf = new byte[1024];
int len;
while ((len = in.read(buf)) > 0) {
out.write(buf, 0, len);
}
}
}
}
finally
.
Кроме того, вы можете использовать FileChannel для копирования файла. Это может быть быстрее, чем метод байтового копирования при копировании большого файла. Вы не можете использовать его, если ваш файл больше 2 ГБ.
public void copy(File src, File dst) throws IOException {
FileInputStream inStream = new FileInputStream(src);
FileOutputStream outStream = new FileOutputStream(dst);
FileChannel inChannel = inStream.getChannel();
FileChannel outChannel = outStream.getChannel();
inChannel.transferTo(0, inChannel.size(), outChannel);
inStream.close();
outStream.close();
}
java.io.FileNotFoundException: /sdcard/AppProj/IMG_20150626_214946.jpg: open failed: ENOENT (No such file or directory)
на FileOutputStream outStream = new FileOutputStream(dst);
этапе. По тексту я понимаю, что файл не существует, поэтому я проверяю его и dst.mkdir();
при необходимости вызываю , но это все равно не помогает. Я тоже пытался проверить, dst.canWrite();
и он вернулся false
. Может ли это источник проблемы? И да, у меня есть <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
.
try ( FileInputStream inStream = new FileInputStream(src); FileOutputStream outStream = new FileOutputStream(dst) ) {
onProgressUpdate
, чтобы я мог показать его в ProgressBar? В принятом решении я могу вычислить прогресс в цикле while, но не вижу, как это сделать здесь.
Удлинение Kotlin для этого
fun File.copyTo(file: File) {
inputStream().use { input ->
file.outputStream().use { output ->
input.copyTo(output)
}
}
}
contentResolver.openInputStream(uri)
.
Это работало хорошо для меня
public static void copyFileOrDirectory(String srcDir, String dstDir) {
try {
File src = new File(srcDir);
File dst = new File(dstDir, src.getName());
if (src.isDirectory()) {
String files[] = src.list();
int filesLength = files.length;
for (int i = 0; i < filesLength; i++) {
String src1 = (new File(src, files[i]).getPath());
String dst1 = dst.getPath();
copyFileOrDirectory(src1, dst1);
}
} else {
copyFile(src, dst);
}
} catch (Exception e) {
e.printStackTrace();
}
}
public static void copyFile(File sourceFile, File destFile) throws IOException {
if (!destFile.getParentFile().exists())
destFile.getParentFile().mkdirs();
if (!destFile.exists()) {
destFile.createNewFile();
}
FileChannel source = null;
FileChannel destination = null;
try {
source = new FileInputStream(sourceFile).getChannel();
destination = new FileOutputStream(destFile).getChannel();
destination.transferFrom(source, 0, source.size());
} finally {
if (source != null) {
source.close();
}
if (destination != null) {
destination.close();
}
}
}
Это просто на Android O (API 26), как вы видите:
@RequiresApi(api = Build.VERSION_CODES.O)
public static void copy(File origin, File dest) throws IOException {
Files.copy(origin.toPath(), dest.toPath());
}
Возможно, уже слишком поздно для ответа, но самый удобный способ
FileUtils
«s
static void copyFile(File srcFile, File destFile)
например, это то, что я сделал
`
private String copy(String original, int copyNumber){
String copy_path = path + "_copy" + copyNumber;
try {
FileUtils.copyFile(new File(path), new File(copy_path));
return copy_path;
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
`
Намного проще теперь с Kotlin:
File("originalFileDir", "originalFile.name")
.copyTo(File("newFileDir", "newFile.name"), true)
true
или false
для перезаписи файла назначения
https://kotlinlang.org/api/latest/jvm/stdlib/kotlin.io/java.io.-file/copy-to.html
Вот решение, которое фактически закрывает потоки ввода / вывода, если при копировании возникает ошибка. Это решение использует методы Apache Commons IO IOUtils для копирования и обработки закрытия потоков.
public void copyFile(File src, File dst) {
InputStream in = null;
OutputStream out = null;
try {
in = new FileInputStream(src);
out = new FileOutputStream(dst);
IOUtils.copy(in, out);
} catch (IOException ioe) {
Log.e(LOGTAG, "IOException occurred.", ioe);
} finally {
IOUtils.closeQuietly(out);
IOUtils.closeQuietly(in);
}
}