Как совместить пути в Java?

Есть ли эквивалент Java для System.IO.Path.Combine() в C # /.СЕТЬ? Или какой-либо код для этого?

Этот статический метод объединяет одну или несколько строк в путь.


person Community    schedule 05.01.2009    source источник
comment
Это ТАК вопрос может помочь.   -  person Jim Blizard    schedule 05.01.2009


Ответы (12)


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

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

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

Если вам нужно обслуживать среды до Java-7, вы можете использовать _ 5_, вот так:

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();
}
person Jon Skeet    schedule 05.01.2009
comment
Остерегайтесь абсолютных путей. Версия .NET вернет path2 (игнорируя path1), если path2 - абсолютный путь. Версия Java отбрасывает начальные / или \ и рассматривает их как относительный путь. - person finnw; 12.02.2010
comment
Чтобы удовлетворить @finnw [важный] комментарий, вы можете добавить File f2 = new File(path2); if (f2.isAbsolute()) return f2; где-нибудь в своей функции. - person Piotr Findeisen; 10.05.2011
comment
Это неэффективно и нечисто, так как требует создания объекта File, а затем возврата к строке. - person WhyNotHugo; 07.03.2012
comment
@ Хьюго: Значит, он тратит целиком два объекта? Шокирует! Честно говоря, для меня это выглядит довольно чисто ... он сохраняет логику для относительных имен файлов, которым он принадлежит, в классе File. - person Jon Skeet; 07.03.2012
comment
тем не менее, я бы сказал, что конструктор new File(String... pathElements) был бы чище, возможно с добавлением new File(File basepath, String... pathElements) - person Martijn; 18.06.2013
comment
новый файл (a / b / c, ../c2) дает a / b / c /../ c2. Как мне вместо этого получить a / b / c2? - person bohdan_trotsenko; 08.10.2013
comment
@modosansreves: Посмотрите на File.getCanonicalPath. - person Jon Skeet; 08.10.2013
comment
@SargeBorsch: Ну, C # - это просто язык. Вы можете легко создать свой собственный эквивалент File на C #, если захотите. (Я предполагаю, что вы имеете в виду, что существование File является преимуществом, с которым я согласен.) - person Jon Skeet; 14.03.2014
comment
@SargeBorsch: В наши дни Java работает больше с Path объектами, чем с _2 _..., но в .NET всегда есть FileInfo и DirectoryInfo, если хотите. Тем не менее, это не вопрос C #, учитывая, что это всего лишь язык ... - person Jon Skeet; 14.10.2014
comment
Использование объектов для имен файлов излишне беспорядочно. Доказательства: попытки работать с ними в PowerShell. Мне постоянно приходится преобразовывать в строки, чтобы передать их какому-нибудь инструменту командной строки, и в итоге я получаю кучу котловых пластин, просто пытающихся заставить их выплюнуть правильную строку, тем более, что существует множество разных типов. которые можно использовать для файловых объектов. Раньше я думал, что тебе тоже следует использовать объект, но я никогда не видел настоящего выигрыша. - person jpmc26; 03.01.2015
comment
@ jpmc26: Это только свидетельство того, что это неудобно в определенных условиях. Я нашел это очень полезным в Java для множества других нужд, которые не требуют значения в виде строки. - person Jon Skeet; 03.01.2015

В Java 7 следует использовать _ 1_:

Path newPath = path.resolve(childPath);

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

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

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

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

Сводная функция для resolve - превосходная _ 10_:

Path childPath = path.relativize(newPath);
person Aleksandr Dubinsky    schedule 19.11.2013

Основной ответ - использовать объекты File. Однако Commons IO действительно имеет класс FilenameUtils, которые могут делать такие вещи, например concat ().

person JodaStephen    schedule 05.01.2009
comment
FilenameUtils.concat, если быть точным. commons.apache.org/proper/commons-io/apidocs/org/apache/commons/. - person MikeFHay; 18.04.2013
comment
Если вы работаете с чем-то вроде JSF, вы определенно хотите, чтобы он был основан на строках, поскольку все пути, которые вы получите, будут основаны на строках. - person DanielK; 14.07.2015

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

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

Использование

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();
    }
}
person isNaN1247    schedule 05.02.2012

независимый от платформы подход (использует 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
person Maksim Kostromin    schedule 09.08.2017

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

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

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

person alpian    schedule 14.11.2010

Если вам не нужно больше строк, вы можете использовать com.google.common.io.Files

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

получить

"some/prefix/with/extra/slashes/file/name"
person Gismo Ranas    schedule 03.04.2017

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

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
person Stephan Henningsen    schedule 12.02.2019

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

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();
}
person Bill    schedule 07.12.2010

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

Path file = Paths.get("Some path");
file = Paths.get(file + "Some other path");
person rootExplorr    schedule 16.06.2019

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

    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"

person JackCid    schedule 04.10.2019

Предполагая, что все заданные пути являются абсолютными. вы можете следовать приведенным ниже фрагментам, чтобы объединить эти пути.

String baseURL = "\\\\host\\testdir\\";
String absoluteFilePath = "\\\\host\\testdir\\Test.txt";;
String mergedPath = Paths.get(baseURL, absoluteFilePath.replaceAll(Matcher.quoteReplacement(baseURL), "")).toString();

выходной путь - \\ host \ testdir \ Test.txt.

person U_R_Naveen UR_Naveen    schedule 08.06.2021