C++中如何从路径字符串中获取文件名!

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++中如何从路径字符串中获取文件名!相关的知识,希望对你有一定的参考价值。

如题,最好是能直接从标准库的字符串中获取,实在没有其他的也行...

如图,路径字符串,一个std::string path;怎么冲path中获取 world或者world.shp

C风格:

char *p = strrchr(path.c_str(), '/')

p是path里最后一个'/'的地址。然后

string s(p + 1);

,s就是"world.shp"了。


C++风格:

int pos = path.find_last_of('/');

pos就是最后一个'/'的下标。

然后

string s(path.substr(pos + 1) );

s就是"world.shp"了。

参考技术A //将字符串根据给定分隔符分割成多个子字符串
//strLine 要分割的字符串
//cDelimiter 分割字符
//strArray string数组用来存放分割好的字符串
int ParseSubStrByDelimiter(string strLine, char cDelimiter, string* strArray)

    int nSeparator = -1;
    int nNextSeparator = -1;
    int nStrLen = strLine.length();
    int nStrArraySize = 0;
    int i = 0;
    
    //strLine replace '\\t' to ' '
int nIndex = strLine.find('\\t', 0);
while(nIndex >= 0)

strLine.replace(nIndex, 1, " ");
nIndex = strLine.find('\\t', 0);


int nCommaPos = strLine.find(',');
    int nSemicolonPos = strLine.find(';');
    if (nCommaPos < 0 && nSemicolonPos < 0)
    
        cDelimiter = '/';
    
 
    switch (cDelimiter)
    
    case '/ ':
        for (i = 0; i < nStrLen;)
        
            if (strLine.at(i) != cDelimiter)
            
                nSeparator = i;
                nNextSeparator = strLine.find(cDelimiter, nSeparator + 1);
                if (nNextSeparator < 0)
                
                    strArray[nStrArraySize++] = strLine.substr(nSeparator, nStrLen - 1);
                    break;
                
                strArray[nStrArraySize++] = strLine.substr(nSeparator, nNextSeparator-nSeparator);
                
                i = nNextSeparator;
                if (i >= nStrLen)
                
                    break;
                
            
            else
            
                i++;
            
        
        break;
        
    case ',':
    case ';':
        for (i = 0; i < nStrLen;)
        
            if (i == 0)
            
                if (strLine.at(i) == cDelimiter)
                
                    strArray[nStrArraySize++] = "";
                    nSeparator = i;
                
                else
                
                    nSeparator = strLine.find(cDelimiter, nSeparator + 1);
                    if (nSeparator < 0)
                    
                        break;
                    
                    strArray[nStrArraySize++] = strLine.substr(0, nSeparator);
                
                
                i = nSeparator;
            
            else
            
                nSeparator = i;
                nNextSeparator = strLine.find(cDelimiter, nSeparator + 1);
                if (nNextSeparator < 0)
                
                    strArray[nStrArraySize++] = strLine.substr(nSeparator + 1, nStrLen - nSeparator - 1);
                    break;
                
                
                if (nNextSeparator == nSeparator + 1)
                
                    strArray[nStrArraySize++] = "";
                
                else
                
                    strArray[nStrArraySize++] = strLine.substr(nSeparator + 1, nNextSeparator - nSeparator - 1);
                
                
                i = nNextSeparator;
            
        
        break;
    default:
        break;
    
    return nStrArraySize;

对于你这个字符串而言这么用

string strArray[15];

int Arraysize= ParseSubStrByDelimiter(strline,'/',strArray);

strArray[Arraysize-1]就是你要的

参考技术B PathFindFileName("D:\OSGEarth2.4/data/world.shp") 可以直接返回 world.shp 参考技术C find_last_of("/") + substring()

如何从包含绝对文件路径的字符串中获取文件名?

【中文标题】如何从包含绝对文件路径的字符串中获取文件名?【英文标题】:How do I get the file name from a String containing the Absolute file path? 【发布时间】:2013-01-09 16:18:54 【问题描述】:

String 变量包含一个文件名C:\Hello\AnotherFolder\The File Name.PDF。如何只获取文件名The File Name.PDF 作为字符串?

我打算拆分字符串,但这不是最佳解决方案。

【问题讨论】:

参考issuetracker.google.com/issues/37131215,已修复。 【参考方案1】:

只需使用 File.getName()

File f = new File("C:\\Hello\\AnotherFolder\\The File Name.PDF");
System.out.println(f.getName());

使用字符串方法

  File f = new File("C:\\Hello\\AnotherFolder\\The File Name.PDF");  
System.out.println(f.getAbsolutePath().substring(f.getAbsolutePath().lastIndexOf("\\")+1));

【讨论】:

小心使用“/”而不是“\”的路径名 @golimar,这应该删除 Windows 和 Unix 路径(不使用 File 类):YourStringPath.replaceAll("^.*[\\/\\\\]", "")【参考方案2】:

使用Path(Java 7+)的替代方法:

Path p = Paths.get("C:\\Hello\\AnotherFolder\\The File Name.PDF");
String file = p.getFileName().toString();

请注意,在\\ 上拆分字符串取决于平台,因为文件分隔符可能会有所不同。 Path#getName 会为您解决这个问题。

【讨论】:

有没有人对这个问题的各种方法进行过性能比较? @crush 我认为Paths.get 不会访问文件系统,因此不会期望性能与 substring/indexOf 有很大不同。 安卓上怎么没有?很奇怪。 是的,Path 处理斜杠/反斜杠的平台相关问题,但前提是文件路径来自同一台机器(或平台)。考虑一下:您从Internet Explorer 上传文件,它的路径为"C:\\Hello\\AnotherFolder\\The File Name.PDF",但您的代码在Unix/Linux 机器上运行,然后p.getFileName() 将返回整个路径,而不仅仅是The File Name.PDF 打电话给toString()好尴尬。【参考方案3】:

在Apache Commons IO 中使用FilenameUtils

String name1 = FilenameUtils.getName("/ab/cd/xyz.txt");
String name2 = FilenameUtils.getName("c:\\ab\\cd\\xyz.txt");

【讨论】:

我觉得这个可能是最好的,因为有时候你可能需要处理其他平台的文件路径 这是最好的,因为:易于阅读,最健壮,代码中的对象更少 +1 顺便说一句,任何想要获取文件名减去扩展名的人都可以使用FilenameUtils.getBaseName(filePath) 这是最好的解决方案。它独立于平台。【参考方案4】:

考虑到您要询问的String

C:\Hello\AnotherFolder\The File Name.PDF

我们需要提取最后一个分隔符之后的所有内容,即。 \。这就是我们感兴趣的。

你可以的

String fullPath = "C:\\Hello\\AnotherFolder\\The File Name.PDF";
int index = fullPath.lastIndexOf("\\");
String fileName = fullPath.substring(index + 1);

这将检索String 中最后一个\ 的索引,并将其后的所有内容提取到fileName

如果您有一个带有不同分隔符的 String,请调整 lastIndexOf 以使用该分隔符。(甚至还有一个 overload 接受整个 String 作为分隔符。 )

我在上面的例子中省略了它,但是如果你不确定String 来自哪里或者它可能包含什么,你会想要验证lastIndexOf 返回一个非负值,因为Javadoc states 它会返回

-1 如果没有出现这种情况

【讨论】:

【参考方案5】:

从 1.7 开始

    Path p = Paths.get("c:\\temp\\1.txt");
    String fileName = p.getFileName().toString();
    String directory = p.getParent().toString();

【讨论】:

【参考方案6】:

您可以使用 path = C:\Hello\AnotherFolder\TheFileName.PDF

String strPath = path.substring(path.lastIndexOf("\\")+1, path.length());

【讨论】:

你应该使用 \\ 而不是 \ 你不应该使用其中的任何一个,因为它依赖于平台。 / on unix 和 \`(AND THERE IS A BUG IN THE MARKDOWN PARSER HERE) on windows. You can't know. Use another solution like File` 或 Paths File.separator 也依赖于平台吗?或者这行得通吗...String strPath = path.substring(path.lastIndexOf(File.separator)+1, path.length()); File.separator & File.separatorChar 在 JDK 的 UNIX/Linux/macOS 版本上都是“/”,在 Windows 版本上都是“\”。 File.separator 在这里并不总是有效,因为在 Windows 中,文件名可以用 "/""\\" 分隔。【参考方案7】:

其他答案对我的具体情况不太适用,我正在阅读源自与我当前操作系统不同的操作系统的路径。为了详细说明,我将从 Windows 平台保存的电子邮件附件保存在 Linux 服务器上。从 JavaMail API 返回的文件名类似于 'C:\temp\hello.xls'

我最终得到的解决方案:

String filenameWithPath = "C:\\temp\\hello.xls";
String[] tokens = filenameWithPath.split("[\\\\|/]");
String filename = tokens[tokens.length - 1];

【讨论】:

【参考方案8】:

考虑 Java 是多平台的情况:

int lastPath = fileName.lastIndexOf(File.separator);
if (lastPath!=-1)
    fileName = fileName.substring(lastPath+1);

【讨论】:

【参考方案9】:

一种没有任何依赖关系的方法,负责处理 ... 和重复的分隔符。

public static String getFileName(String filePath) 
    if( filePath==null || filePath.length()==0 )
        return "";
    filePath = filePath.replaceAll("[/\\\\]+", "/");
    int len = filePath.length(),
        upCount = 0;
    while( len>0 ) 
        //remove trailing separator
        if( filePath.charAt(len-1)=='/' ) 
            len--;
            if( len==0 )
                return "";
        
        int lastInd = filePath.lastIndexOf('/', len-1);
        String fileName = filePath.substring(lastInd+1, len);
        if( fileName.equals(".") ) 
            len--;
        
        else if( fileName.equals("..") ) 
            len -= 2;
            upCount++;
        
        else 
            if( upCount==0 )
                return fileName;
            upCount--;
            len -= fileName.length();
        
    
    return "";

测试用例:

@Test
public void testGetFileName() 
    assertEquals("", getFileName("/"));
    assertEquals("", getFileName("////"));
    assertEquals("", getFileName("//C//.//../"));
    assertEquals("", getFileName("C//.//../"));
    assertEquals("C", getFileName("C"));
    assertEquals("C", getFileName("/C"));
    assertEquals("C", getFileName("/C/"));
    assertEquals("C", getFileName("//C//"));
    assertEquals("C", getFileName("/A/B/C/"));
    assertEquals("C", getFileName("/A/B/C"));
    assertEquals("C", getFileName("/C/./B/../"));
    assertEquals("C", getFileName("//C//./B//..///"));
    assertEquals("user", getFileName("/user/java/.."));
    assertEquals("C:", getFileName("C:"));
    assertEquals("C:", getFileName("/C:"));
    assertEquals("java", getFileName("C:\\Program Files (x86)\\java\\bin\\.."));
    assertEquals("C.ext", getFileName("/A/B/C.ext"));
    assertEquals("C.ext", getFileName("C.ext"));

也许 getFileName 有点令人困惑,因为它也返回目录名称。它返回路径中文件或最后一个目录的名称。

【讨论】:

【参考方案10】:

java.nio.file.Path 的getFileName() 方法用于返回此路径对象指向的文件或目录的名称。

路径 getFileName()

供参考:

https://www.geeksforgeeks.org/path-getfilename-method-in-java-with-examples/

【讨论】:

【参考方案11】:

使用 java regex * 提取文件名。

public String extractFileName(String fullPathFile)
        try 
            Pattern regex = Pattern.compile("([^\\\\/:*?\"<>|\r\n]+$)");
            Matcher regexMatcher = regex.matcher(fullPathFile);
            if (regexMatcher.find())
                return regexMatcher.group(1);
            
         catch (PatternSyntaxException ex) 
            LOG.info("extractFileName::pattern problem <"+fullPathFile+">",ex);
        
        return fullPathFile;
    

【讨论】:

【参考方案12】:

您可以使用 FileInfo 对象来获取文件的所有信息。

    FileInfo f = new FileInfo(@"C:\Hello\AnotherFolder\The File Name.PDF");
    MessageBox.Show(f.Name);
    MessageBox.Show(f.FullName);
    MessageBox.Show(f.Extension );
    MessageBox.Show(f.DirectoryName);

【讨论】:

【参考方案13】:

这个答案在 c# 中对我有用:

using System.IO;
string fileName = Path.GetFileName("C:\Hello\AnotherFolder\The File Name.PDF");

【讨论】:

以上是关于C++中如何从路径字符串中获取文件名!的主要内容,如果未能解决你的问题,请参考以下文章

如何快速从文件路径中获取文件名

如何从包含 android 中的绝对文件路径的字符串中获取文件夹名称?

如何从路径字符串中获取最后一个文件夹?

C++ Windows - 通过可执行文件路径获取进程的 PID

如何从文件的完整路径中获取目录?

如何在 C++ 中找到从 ofstream 创建的文件的目录路径?