parent_path() 带或不带斜杠
parent_path() with or without trailing slash
如 documentation 中所述,以下的预期输出是:
boost::filesystem::path filePath1 = "/home/user/";
cout << filePath1.parent_path() << endl; // outputs "/home/user"
boost::filesystem::path filePath2 = "/home/user";
cout << filePath2.parent_path() << endl; // outputs "/home"
问题是,你是怎么处理的?也就是说,如果我接受一个路径作为参数,我不希望用户关心它是否应该有尾部斜线。看起来最简单的做法是在尾部附加一个斜杠,然后调用 parent_path()
两次以获得我想要的“/home”的父路径:
boost::filesystem::path filePath1 = "/home/user/";
filePath1 /= "/";
cout << filePath1.parent_path().parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath2 = "/home/user";
filePath2 /= "/";
cout << filePath2.parent_path().parent_path() << endl; // outputs "/home"
但这似乎很荒谬。有没有更好的方法在框架内处理这个问题?
似乎是这样,尽管我建议对目录 string
进行先前的操作,而不是调用两次 parent_path()
:
std::string directory = "/home/user"; // Try with "/home/user/" too, result is the same
while ((directory.back() == '/') || (directory.back() == '\')))
directory.erase(directory.size()-1);
boost::filesystem::path filePath(directory);
std::cout << filePath.parent_path() << std::endl; // outputs "/home"
重要的是要注意 std::string::back()
是 C++11 的特性。如果您需要使用以前的版本进行编译,则必须稍微更改算法。
有一个(未记录?)成员函数:path& path::remove_trailing_separator();
我试过了,它在 Windows 上对我有用 1.60.0
:
boost::filesystem::path filePath1 = "/home/user/";
cout << filePath1.parent_path() << endl; // outputs "/home/user"
cout << filePath1.remove_trailing_separator().parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath2 = "/home/user";
cout << filePath2.parent_path() << endl; // outputs "/home"
cout << filePath2.remove_trailing_separator().parent_path() << endl; // outputs "/home"
要从指向目录的路径中删除尾随分隔符,到目前为止这对我有用:
/**
* Creates lexically normal (removes extra path separators and dots) directory
* path without trailing path separator slash(es)
* @param dir_path - directory path to normalize
*/
void normalize_dir_path(boost::filesystem::path& dir_path) {
// @HACK - append non-existing file to path so that we may later resolve
// normalized directory path using parent_path()
dir_path /= "FILE.TXT";
// Remove unneeded dots and slashes
dir_path = dir_path.lexically_normal();
// Remove trailing slash from original path!
dir_path = dir_path.parent_path();
}
以上答案类似于 OP 最初发布的解决方法(添加 '/'
)以及 Wurmloch 关于使用 lexically_normal()
的评论。一个优点是只使用 boost::filesystem
中记录的方法。一个可能的缺点是调用者必须确信输入参数 dir_path
是目录而不是常规文件。
使用normalize_dir_path(...)
方法回答OP的问题:
boost::filesystem::path filePath1 = "/home/user/";
normalize_dir_path(filePath1); // filePath1 is now "/home/user"
cout << filePath1.parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath2 = "/home/user";
normalize_dir_path(filePath2); // filePath2 is now "/home/user"
cout << filePath2.parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath3 = "/home/user/.";
normalize_dir_path(filePath3); // filePath3 is now "/home/user"
cout << filePath3.parent_path() << endl; // outputs "/home"
更新
刚刚意识到 boost::filesystem::path::lexically_normal()
仅适用于 BOOST 版本 >= 1_60_0。对于早期版本,默认情况下似乎有一个已弃用的函数可用 boost::filesystem::path::normalize()
(只要未定义 BOOST_FILESYSTEM_NO_DEPRECATED
)。所以,我目前规范化目录路径的方法是这样的:
#include <boost/version.hpp>
void normalize_dir_path(boost::filesystem::path& dir_path) {
// @HACK - append non-existing file to path so that we may later resolve
// normalized directory path using parent_path()
dir_path /= "FILE.TXT";
// Remove unneeded dots and slashes
#if BOOST_VERSION >= 106000
dir_path = dir_path.lexically_normal();
#else
dir_path.normalize();
#endif
// Remove trailing slash from original path!
dir_path = dir_path.parent_path();
}
您可以在 C++17 中使用 std::filesystem::canonical
:
namespace fs = std::filesystem;
fs::path tmp = "c:\temp\";
tmp = fs::canonical(tmp); // will remove slash
fs::path dir_name = tmp.filename(); // will get temp
如 documentation 中所述,以下的预期输出是:
boost::filesystem::path filePath1 = "/home/user/";
cout << filePath1.parent_path() << endl; // outputs "/home/user"
boost::filesystem::path filePath2 = "/home/user";
cout << filePath2.parent_path() << endl; // outputs "/home"
问题是,你是怎么处理的?也就是说,如果我接受一个路径作为参数,我不希望用户关心它是否应该有尾部斜线。看起来最简单的做法是在尾部附加一个斜杠,然后调用 parent_path()
两次以获得我想要的“/home”的父路径:
boost::filesystem::path filePath1 = "/home/user/";
filePath1 /= "/";
cout << filePath1.parent_path().parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath2 = "/home/user";
filePath2 /= "/";
cout << filePath2.parent_path().parent_path() << endl; // outputs "/home"
但这似乎很荒谬。有没有更好的方法在框架内处理这个问题?
似乎是这样,尽管我建议对目录 string
进行先前的操作,而不是调用两次 parent_path()
:
std::string directory = "/home/user"; // Try with "/home/user/" too, result is the same
while ((directory.back() == '/') || (directory.back() == '\')))
directory.erase(directory.size()-1);
boost::filesystem::path filePath(directory);
std::cout << filePath.parent_path() << std::endl; // outputs "/home"
重要的是要注意 std::string::back()
是 C++11 的特性。如果您需要使用以前的版本进行编译,则必须稍微更改算法。
有一个(未记录?)成员函数:path& path::remove_trailing_separator();
我试过了,它在 Windows 上对我有用 1.60.0
:
boost::filesystem::path filePath1 = "/home/user/";
cout << filePath1.parent_path() << endl; // outputs "/home/user"
cout << filePath1.remove_trailing_separator().parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath2 = "/home/user";
cout << filePath2.parent_path() << endl; // outputs "/home"
cout << filePath2.remove_trailing_separator().parent_path() << endl; // outputs "/home"
要从指向目录的路径中删除尾随分隔符,到目前为止这对我有用:
/**
* Creates lexically normal (removes extra path separators and dots) directory
* path without trailing path separator slash(es)
* @param dir_path - directory path to normalize
*/
void normalize_dir_path(boost::filesystem::path& dir_path) {
// @HACK - append non-existing file to path so that we may later resolve
// normalized directory path using parent_path()
dir_path /= "FILE.TXT";
// Remove unneeded dots and slashes
dir_path = dir_path.lexically_normal();
// Remove trailing slash from original path!
dir_path = dir_path.parent_path();
}
以上答案类似于 OP 最初发布的解决方法(添加 '/'
)以及 Wurmloch 关于使用 lexically_normal()
的评论。一个优点是只使用 boost::filesystem
中记录的方法。一个可能的缺点是调用者必须确信输入参数 dir_path
是目录而不是常规文件。
使用normalize_dir_path(...)
方法回答OP的问题:
boost::filesystem::path filePath1 = "/home/user/";
normalize_dir_path(filePath1); // filePath1 is now "/home/user"
cout << filePath1.parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath2 = "/home/user";
normalize_dir_path(filePath2); // filePath2 is now "/home/user"
cout << filePath2.parent_path() << endl; // outputs "/home"
boost::filesystem::path filePath3 = "/home/user/.";
normalize_dir_path(filePath3); // filePath3 is now "/home/user"
cout << filePath3.parent_path() << endl; // outputs "/home"
更新
刚刚意识到 boost::filesystem::path::lexically_normal()
仅适用于 BOOST 版本 >= 1_60_0。对于早期版本,默认情况下似乎有一个已弃用的函数可用 boost::filesystem::path::normalize()
(只要未定义 BOOST_FILESYSTEM_NO_DEPRECATED
)。所以,我目前规范化目录路径的方法是这样的:
#include <boost/version.hpp>
void normalize_dir_path(boost::filesystem::path& dir_path) {
// @HACK - append non-existing file to path so that we may later resolve
// normalized directory path using parent_path()
dir_path /= "FILE.TXT";
// Remove unneeded dots and slashes
#if BOOST_VERSION >= 106000
dir_path = dir_path.lexically_normal();
#else
dir_path.normalize();
#endif
// Remove trailing slash from original path!
dir_path = dir_path.parent_path();
}
您可以在 C++17 中使用 std::filesystem::canonical
:
namespace fs = std::filesystem;
fs::path tmp = "c:\temp\";
tmp = fs::canonical(tmp); // will remove slash
fs::path dir_name = tmp.filename(); // will get temp