Як поєднати шляхи на Java?


Відповіді:


407

Замість того, щоб зберігати все на основі рядків, слід використовувати клас, який призначений для представлення шляху до файлової системи.

Якщо ви використовуєте 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
@Matthew - тому що каталог - це файл. Це файл, вміст якого визначає дітей цього редактора, їхнє розташування на диску, дозволи та ін.
Dónal

7
@Hugo: Значить, вона витрачає цілі два об'єкти ? Шокуюче! Якщо мені чесно, то це виглядає досить чисто, але вона зберігає логіку щодо відносних імен файлів, де вони належать, у класі File.
Джон Скіт

1
@modosansreves: Подивіться File.getCanonicalPath.
Джон Скіт

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

118

У Java 7 слід використовувати resolve:

Path newPath = path.resolve(childPath);

Хоча клас NIO2 Path може здатися трохи зайвим для File з непотрібним інтерфейсом API, він насправді тонко більш елегантний та надійний.

Зауважте, що Paths.get()(як це запропонував хтось інший) не виникає перевантаження Path, і робити Paths.get(path.toString(), childPath)це НЕ те саме, що resolve(). З Paths.get()документів :

Зауважте, що хоча цей метод дуже зручний, його використання буде означати припущену посилання на FileSystem за замовчуванням і обмежує корисність викличного коду. Отже, його не слід використовувати в бібліотечному коді, призначеному для гнучкого повторного використання. Більш гнучка альтернатива - використовувати наявний екземпляр Path в якості якоря, наприклад:

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

Сестринська функція resolve- відмінна relativize:

Path childPath = path.relativize(newPath);

42

Основна відповідь - використання об'єктів File. Однак у Commons IO є клас FilenameUtils, який може робити такі речі, як метод concat () .


7
FilenameUtils.concat, якщо бути точним. commons.apache.org/proper/commons-io/apidocs/org/apache/commons/…
MikeFHay

Якщо ви працюєте з чимось на зразок JSF, ви однозначно хочете зберегти його на базі String, оскільки всі Шляхи, які ви отримаєте, будуть засновані на String.
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, AOche 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();
}

2

Якщо вам не потрібно більше рядків, ви можете використовувати com.google.common.io.Files

Files.simplifyPath("some/prefix/with//extra///slashes" + "file//name")

отримати

"some/prefix/with/extra/slashes/file/name"

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.