如何使用GNU C库版本的basename()
和dirname()
?
如果你
#include <libgen.h>
目录名您已经获得了POSIX版本的basename()
,而不是GNU版本。(即使你
#define _GNU_SOURCE
据我所知,在c中没有条件导入。是否有gcc特定的技巧?
你自己写,给它一个不同于basename
的名字。这种GNU坚持创建可在1-3行中编写的标准函数的替代不符合版本的做法完全是愚蠢的。
char *gnu_basename(char *path)
{
char *base = strrchr(path, '/');
return base ? base+1 : path;
}
根据手册页,您应该这样做
#define _GNU_SOURCE
#include <string.h>
如果您获得的是POSIX版本,libgen.h可能在此之前已经包含在其中。您可能希望在CPPFLAGS中包含-D_GNU_SOURCE
用于编译:
gcc -D_GNU_SOURCE ....
在编译器资源管理器上比较POSIX版本和GNU版本
检查libgen.h
后,我很确定我有一个无警告和无错误的解决方案:
/* my C program */
#define _GNU_SOURCE /* for GNU version of basename(3) */
#include <libgen.h> /* for dirname(3) */
#undef basename /* (snide comment about libgen.h removed) */
#include <string.h> /* for basename(3) (GNU version) and strcmp(3) */
/* rest of C program... */
加上#undef
行,现在我的程序包括来自libgen.h
的dirname(3)
和来自string.h
的basename(3)
的GNU版本。
gcc
(4.5.2版本)或clang
(3.3版本)没有编译器警告/错误。
确保使用GNU C库构建,而不是使用系统(假定的)posix兼容的默认库。
通常在GCC规范文件中设置。使用-v选项显示当前设置:
$ gcc -v
Using built-in specs.
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.4.4-14ubuntu5' --with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.4 --enable-shared --enable-multiarch --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.4.5 (Ubuntu/Linaro 4.4.4-14ubuntu5)
太疯狂了,basename和dirname有两个版本。
我们在一个大项目中工作,看起来这两个api已经引起了潜在的bug。因此,我们将"basename" "dirname"标记为deprecated以警告如果有人用它:
#ifdef basename
__attribute__ ((deprecated))
char *__xpg_basename(char *path);
#else
__attribute__ ((deprecated))
char *basename(const char *path);
#endif
__attribute__ ((deprecated))
char *dirname(char *path);
我们还尝试引入一个base c基础库,如glib或libcork,但是它看起来太重了。因此,我们为此编写了一个小库实现如下:
#include <libgen.h> // for dirname
#include <linux/limits.h> // for PATH_MAX
#include <stdio.h> // for snprintf
#include <string.h> // for basename
#include <stdbool.h> // for bool
bool get_basename(const char *path, char *name, size_t name_size) {
char path_copy[PATH_MAX] = {' '};
strncpy(path_copy, path, sizeof(path_copy) - 1);
return snprintf(name, name_size, "%s", basename(path_copy)) < name_size;
}
bool get_dirname(const char *path, char *name, size_t name_size) {
char path_copy[PATH_MAX] = {' '};
strncpy(path_copy, path, sizeof(path_copy) - 1);
return snprintf(name, name_size, "%s", dirname(path_copy)) < name_size;
}
然后我们将所有basename
dirname
调用替换为get_basename
get_dirname
。