Java:如何获取路径的各个部分

时间:2011-06-08 13:07:04

标签: java algorithm filepath

这应该相当简单,但我只是卡住了。假设您拥有路径/a/b/c/。我想将其转换为包含以下内容的数组:

  • /
  • /a/
  • /a/b/
  • /a/b/c/

开头和结尾的斜线应该是可选的。有人帮忙吗?

我将把它用于创建目录的函数,我希望它也创建所有缺失的部分,如果例如ab不存在则不会失败


更新:如果可以,我当然会使用File.mkdirs(),但这不在本地文件系统上。这是为了简化与SFTP库的接口,该库只有mkdir方法以字符串的形式获取路径。

6 个答案:

答案 0 :(得分:11)

为什么不使用File.mkdirs()


编辑:根据您的要求不使用File.mkdirs():

我仍然认为使用File作为辅助类更容易:

package com.example.test;

import java.io.File;
import java.util.LinkedList;
import java.util.List;

public class FileSplitter {
    final private File path;

    public List<String> getPathStrings()
    {
        LinkedList<String> list = new LinkedList<String>();
        File p = this.path;
        while (p != null)
        {
            list.addFirst(p.getPath());
            p = p.getParentFile();
        }
        return list;
    }

    public FileSplitter(File path) { this.path = path; }

    public static void main(String[] args) {
        doit(new File("/foo/bar/baz"));
        doit(new File("/bam/biff/boom/pow"));
    }

    private static void doit(File file) {
        for (String s : new FileSplitter(file)
                .getPathStrings())
            System.out.println(s);      
    }
}

在我的机器(Windows)上打印出来:

\
\foo
\foo\bar
\foo\bar\baz
\
\bam
\bam\biff
\bam\biff\boom
\bam\biff\boom\pow

如果您无论如何都需要使用正斜杠,那么我要么使用字符串而不是文件,要么在使用点上执行.replace('\\','/')


最后,这是一种可能对您的最终应用程序更有帮助的方法。

它具有与前一个相同的输出,但有助于控制反转 您可以在其中将自定义mkdir()作为伪Runnable执行,作为步骤传递给路径迭代器:

package com.example.test;

import java.io.File;

public class PathRunner
{
    final private File path;
    public PathRunner(File path) { 
        this.path = path; 
    }

    public interface Step
    {
        public boolean step(File path);
    }

    public boolean run(Step step) 
    {
        return run(step, this.path);
    }
    private boolean run(Step step, File p)
    {
        if (p == null)
            return true;
        else if (!run(step, p.getParentFile()))
            return false;
        else
            return step.step(p);
    }

    /**** test methods ****/

    public static void main(String[] args) {
        doit(new File("/foo/bar/baz"));
        doit(new File("/bam/biff/boom/pow"));
    }
    private static boolean doit(File path) {
        Step step = new Step()
        {
            @Override public boolean step(File path) {
                System.out.println(path);
                return true;
                /* in a mkdir operation, here's where you would call: 

                return yourObject.mkdir(
                    path.getPath().replace('\\', '/')
                );
                 */
            }               
        };
        return new PathRunner(path).run(step);
    }
}

答案 1 :(得分:5)

如果你需要原始的东西。尝试拆分并追加。

public class StackOverflow {

    public static void main(String args[]) {

        String[] folders = "/a/b/c/".split("/");
        String[] paths = new String[folders.length];
        String path = "";

        for (int i = 0; i < folders.length; i++) {
            path +=   folders[i] + "/";
            paths[i] = path;
        }
    }
}

这是代码块的输出:

run:
/
/a/
/a/b/
/a/b/c/
BUILD SUCCESSFUL (total time: 0 seconds)

答案 2 :(得分:4)

File类支持此功能。

public static void main(String... args) {
    split(new File("/a/b/c/d/e"));
    split(new File("\\A\\B\\C\\D\\E"));
}

private static void split(File file) {
    File parent = file.getParentFile();
    if (parent != null) split(parent);
    System.out.println(file);
}
Windows上的

打印

\
\a
\a\b
\a\b\c
\a\b\c\d
\a\b\c\d\e
\
\A
\A\B
\A\B\C
\A\B\C\D
\A\B\C\D\E

答案 3 :(得分:2)

无需这样做。 File.mkdirs() instead

答案 4 :(得分:2)

结束此代码:

   public String[] componizePath(String path)
   {
      ArrayList<String> parts = new ArrayList<String>();  

      int index = 0;
      while(index < path.length())
      {
         if(path.charAt(index) == '/' || index == path.length()-1)
         {
            parts.add(path.substring(0, index+1));
         }
         index++;
      }

      return parts.toArray(new String[0]);
   }

JUnit测试:

   @Test
   public void componizePath_EmptyPath()
   {
      String[] actual = getSftp().componizePath("");
      String[] expected = new String[0];
      assertArrayEquals(expected, actual);
   }

   @Test
   public void componizePath_RootPath()
   {
      String[] actual = getSftp().componizePath("/");
      String[] expected = new String[] {"/"};
      assertArrayEquals(expected, actual);
   }

   @Test
   public void componizePath_SimplePath()
   {
      String[] actual = getSftp().componizePath("a");
      String[] expected = new String[] {"a"};
      assertArrayEquals(expected, actual);
   }

   @Test
   public void componizePath_SimplePathWithTrailingSlash()
   {
      String[] actual = getSftp().componizePath("a/");
      String[] expected = new String[] {"a/"};
      assertArrayEquals(expected, actual);
   }

   @Test
   public void componizePath_ComplexerPath()
   {
      String[] actual = getSftp().componizePath("a/b/cc");
      String[] expected = new String[] {"a/", "a/b/", "a/b/cc"};
      assertArrayEquals(expected, actual);
   }

   @Test
   public void componizePath_ComplexerPathWithTrailingSlash()
   {
      String[] actual = getSftp().componizePath("a/b/c/");
      String[] expected = new String[] {"a/", "a/b/", "a/b/c/"};
      assertArrayEquals(expected, actual);
   }

   @Test
   public void componizePath_ComplexerPathWithLeadingSlash()
   {
      String[] actual = getSftp().componizePath("/a/b/c");
      String[] expected = new String[] {"/", "/a/", "/a/b/", "/a/b/c"};
      assertArrayEquals(expected, actual);
   }

答案 5 :(得分:0)

这应该是一个相当简单的算法,如果给你一个字符串路径=“/ a / b / c /”,你可以执行以下操作:

def f(st):
    index = 0
    array = []
    while index < len(st):
        if(st[index] == '/'):
            array += [st[:index+1]]

        index += 1
    return array

这是一个python算法,您可以轻松地将其转换为Java。当然你可以使用Jason和I82Much指出的File.mkdirs(),但看看算法是如何工作的也很有趣。

修改 在Java的情况下,你不能迭代一个字符串,但你可以将字符串转换为字符串的arraylist,当你迭代时,你写的条件是如果charachter是'/',那么创建一个连接的新字符串从第一个字符到我们找到的当前字符的所有字符应该是'/',然后将它添加到之前初始化的字符串列表中。生成的字符串列表将是您需要的列表。 如果您自己尝试并看看它是如何进行的,那么这对您来说是一种很好的做法。当然,您可以遍历结果列表并创建目录。