Чи існує еквівалент Java для System.IO.Path.Combine()
C # /. NET? Або будь-який код для цього?
Цей статичний метод поєднує одну або кілька рядків у шлях.
Чи існує еквівалент Java для System.IO.Path.Combine()
C # /. NET? Або будь-який код для цього?
Цей статичний метод поєднує одну або кілька рядків у шлях.
Відповіді:
Замість того, щоб зберігати все на основі рядків, слід використовувати клас, який призначений для представлення шляху до файлової системи.
Якщо ви використовуєте 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();
}
path2
(ігноруючи path1
), якщо path2
це абсолютний шлях. Версія Java скине провідну /
або \
трактуватиме її як відносний шлях.
File.getCanonicalPath
.
File
у C #, якщо цього хочете. (Я припускаю, що ви маєте на увазі існування File
користі, з якою я погоджуюся.)
У 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);
Основна відповідь - використання об'єктів File. Однак у Commons IO є клас FilenameUtils, який може робити такі речі, як метод concat () .
Я знаю це вже давно з початкової відповіді Йона, але в мене була схожа вимога до ОП.
Шляхом розширення рішення Джона я придумав наступне, що займе один або кілька сегментів шляху, займає стільки сегментів шляху, які ви можете на нього кинути.
Використання
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();
}
}
незалежний підхід від платформи (використовує 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
Щоб покращити відповідь JodaStephen, AOche Commons IO має FilenameUtils, який робить це. Приклад (в Linux):
assert org.apache.commons.io.FilenameUtils.concat("/home/bob", "work\\stuff.log") == "/home/bob/work/stuff.log"
Це незалежна платформа, і випускатимуть усі роздільники, які вам потрібні.
Ось рішення, яке обробляє декілька частин шляху та крайові умови:
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();
}
Якщо вам не потрібно більше рядків, ви можете використовувати com.google.common.io.Files
Files.simplifyPath("some/prefix/with//extra///slashes" + "file//name")
отримати
"some/prefix/with/extra/slashes/file/name"
Можливо, пізно на вечірку, але я хотів поділитися цим питанням. Я використовую шаблон 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
Це також працює в Java 8:
Path file = Paths.get("Some path");
file = Paths.get(file + "Some other path");
Це рішення пропонує інтерфейс для приєднання фрагментів шляху з масиву 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"