8

如何使用 和 的 GNU C 库basename()版本dirname()

如果你

#include <libgen.h>

对于 dirname 您已经获得了 POSIX,而不是 GNU 版本的basename(). (即使你

#define _GNU_SOURCE

据我所知,C 中没有条件导入。有 gcc 特定的技巧吗?

4

5 回答 5

9

只需自己编写并给它起一个不同于basename. GNU 坚持创建可以用 1-3 行编写的标准函数的替代不合格版本,这完全是愚蠢的。

char *gnu_basename(char *path)
{
    char *base = strrchr(path, '/');
    return base ? base+1 : path;
}

这样,您的程序也将更具可移植性。

于 2011-04-27T13:30:23.570 回答
4

根据你应该做的手册页

      #define _GNU_SOURCE
       #include <string.h>
       #include <libgen.h>

如果您获得 POSIX 版本,则可能在此之前已经包含 libgen.h。您可能希望包含-D_GNU_SOURCE在 CPPFLAGS 中进行编译:

gcc -D_GNU_SOURCE ....
于 2011-04-27T09:59:28.757 回答
1

经过检查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条线,现在我的程序包括dirname(3)fromlibgen.h和 GNU 版本的basename(3)from string.h

gcc(4.5.2 版)或clang(3.3版)没有编译器警告/错误。

于 2014-01-25T22:08:41.470 回答
0

确保您使用的是 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)
于 2011-04-27T09:58:05.947 回答
0

这是疯狂的 basename 和 dirname 有两个版本。

我们在一个大项目中工作,看起来这两个 api 已经引起了潜在的错误。因此,如果有人使用它,我们将“basename”“dirname”标记为已弃用以警告:

#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] = {'\0'};
  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] = {'\0'};
  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.

于 2019-02-20T07:00:06.370 回答