Как объединить пути в Java?


Ответы:


408

Вместо того, чтобы хранить все на основе строк, вы должны использовать класс, который предназначен для представления пути к файловой системе.

Если вы используете Java 7 или Java 8, вам следует рассмотреть возможность использования java.nio.file.Path; Path.resolveможет использоваться для объединения одного пути с другим или со строкой. PathsВспомогательный класс полезен тоже. Например:

Path path = Paths.get("foo", "bar", "baz.txt");

Если вам нужно обслуживать среды до Java-7, вы можете использовать java.io.Fileэто так:

File baseDirectory = new File("foo");
File subDirectory = new File(baseDirectory, "bar");
File fileInDirectory = new File(subDirectory, "baz.txt");

Если вы хотите вернуть его как строку позже, вы можете позвонить getPath(). Действительно, если вы действительно хотите подражать Path.Combine, вы можете просто написать что-то вроде:

public static String combine(String path1, String path2)
{
    File file1 = new File(path1);
    File file2 = new File(file1, path2);
    return file2.getPath();
}

10
Остерегайтесь абсолютных путей. Версия .NET вернет path2(игнорируя path1), если path2это абсолютный путь. Версия Java отбросит начальный /или \ и будет относиться к нему как к относительному пути.
finnw

23
@ Матфея - потому что каталог - это файл. Это файл, содержимое которого определяет дочерние элементы этого каталога, их местоположение на диске, разрешения и т. Д.
Dónal

7
@Hugo: Значит, он тратит целых два объекта ? Возмутительно! Честно говоря, для меня это выглядит довольно чисто ... он сохраняет логику относительных имен файлов в своем классе File.
Джон Скит

1
@modosansreves: посмотрите File.getCanonicalPath.
Джон Скит

1
@SargeBorsch: Ну, C # это просто язык. Вы можете легко создать свой собственный эквивалент Fileв C #, если хотите. (Я предполагаю, что вы имеете в виду, что существование File- это выгода, с которой я бы согласился.)
Джон Скит,

118

В Java 7 вы должны использовать resolve:

Path newPath = path.resolve(childPath);

Хотя класс Path NIO2 может показаться немного избыточным для File с неоправданно другим API, на самом деле он немного элегантнее и надежнее.

Обратите внимание, что Paths.get()(как предполагает кто-то другой) не имеет перегрузки, принимающей a Path, и выполнение Paths.get(path.toString(), childPath)НЕ является тем же, что и resolve(). Из Paths.get()документов :

Обратите внимание, что хотя этот метод очень удобен, его использование подразумевает предполагаемую ссылку на файловую систему по умолчанию и ограничивает полезность вызывающего кода. Следовательно, он не должен использоваться в коде библиотеки, предназначенном для гибкого повторного использования. Более гибкой альтернативой является использование существующего экземпляра Path в качестве привязки, например:

Path dir = ...
Path path = dir.resolve("file");

Сестринская функция resolveявляется превосходной relativize:

Path childPath = path.relativize(newPath);

42

Основной ответ заключается в использовании файловых объектов. Однако у Commons IO действительно есть класс FilenameUtils, который может делать такие вещи, такие как метод concat () .


7
FilenameUtils.concat, если быть точным. commons.apache.org/proper/commons-io/apidocs/org/apache/commons/…
MikeFHay

Если вы работаете с чем-то вроде JSF, вы определенно хотите сохранить его на основе строк, так как все пути, которые вы получите, будут на основе строк.
DanielK

17

Я знаю, что прошло много времени с момента первоначального ответа Джона, но у меня было такое же требование к ОП.

Чтобы расширить решение Джона, я придумал следующее: один или несколько сегментов пути будут занимать столько сегментов пути, сколько вы сможете на них набросать.

Применение

Path.combine("/Users/beardtwizzle/");
Path.combine("/", "Users", "beardtwizzle");
Path.combine(new String[] { "/", "Users", "beardtwizzle", "arrayUsage" });

Код здесь для других с похожей проблемой

public class Path {
    public static String combine(String... paths)
    {
        File file = new File(paths[0]);

        for (int i = 1; i < paths.length ; i++) {
            file = new File(file, paths[i]);
        }

        return file.getPath();
    }
}

12

независимый от платформы подход (использует File.separator, т.е. будет работать, зависит от операционной системы, в которой выполняется код:

java.nio.file.Paths.get(".", "path", "to", "file.txt")
// relative unix path: ./path/to/file.txt
// relative windows path: .\path\to\filee.txt

java.nio.file.Paths.get("/", "path", "to", "file.txt")
// absolute unix path: /path/to/filee.txt
// windows network drive path: \\path\to\file.txt

java.nio.file.Paths.get("C:", "path", "to", "file.txt")
// absolute windows path: C:\path\to\file.txt

11

Чтобы улучшить ответ JodaStephen, Apache Commons IO имеет FilenameUtils, который делает это. Пример (в Linux):

assert org.apache.commons.io.FilenameUtils.concat("/home/bob", "work\\stuff.log") == "/home/bob/work/stuff.log"

Он не зависит от платформы и будет производить любые разделители, которые нужны вашей системе.


2

Вот решение, которое обрабатывает несколько частей пути и краевые условия:

public static String combinePaths(String ... paths)
{
  if ( paths.length == 0)
  {
    return "";
  }

  File combined = new File(paths[0]);

  int i = 1;
  while ( i < paths.length)
  {
    combined = new File(combined, paths[i]);
    ++i;
  }

  return combined.getPath();
}


1

Возможно, поздно вечером, но я хотел поделиться своим мнением по этому поводу. Я использую шаблон Builder и разрешаю удобно соединенные appendвызовы. Его можно легко расширить для поддержки работы с Pathобъектами.

public class Files  {
    public static class PathBuilder {
        private File file;

        private PathBuilder ( File root ) {
            file = root;
        }

        private PathBuilder ( String root ) {
            file = new File(root);
        }

        public PathBuilder append ( File more ) {
            file = new File(file, more.getPath()) );
            return this;
        }

        public PathBuilder append ( String more ) {
            file = new File(file, more);
            return this;
        }

        public File buildFile () {
            return file;
        }
    }

    public static PathBuilder buildPath ( File root ) {
        return new PathBuilder(root);
    }

    public static PathBuilder buildPath ( String root ) {
        return new PathBuilder(root);
    }
}

Пример использования:

File root = File.listRoots()[0];
String hello = "hello";
String world = "world";
String filename = "warez.lha"; 

File file = Files.buildPath(root).append(hello).append(world)
              .append(filename).buildFile();
String absolute = file.getAbsolutePath();

В результате absoluteбудет что-то вроде:

/hello/world/warez.lha

или, может быть, даже:

A:\hello\world\warez.lha

1

Это также работает в Java 8:

Path file = Paths.get("Some path");
file = Paths.get(file + "Some other path");

0

Это решение предлагает интерфейс для объединения фрагментов пути из массива String []. Он использует java.io.File.File (String parent, String child) :

    public static joinPaths(String[] fragments) {
        String emptyPath = "";
        return buildPath(emptyPath, fragments);
    }

    private static buildPath(String path, String[] fragments) {
        if (path == null || path.isEmpty()) {
            path = "";
        }

        if (fragments == null || fragments.length == 0) {
            return "";
        }

        int pathCurrentSize = path.split("/").length;
        int fragmentsLen = fragments.length;

        if (pathCurrentSize <= fragmentsLen) {
            String newPath = new File(path, fragments[pathCurrentSize - 1]).toString();
            path = buildPath(newPath, fragments);
        }

        return path;
    }

Тогда вы можете просто сделать:

String[] fragments = {"dir", "anotherDir/", "/filename.txt"};
String path = joinPaths(fragments);

Возвращает:

"/dir/anotherDir/filename.txt"
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.