我正在尝试获取一组九个 *.c 文件(以及九个相关的 *.h 文件)在 Windows 下进行编译。
该代码最初是在 Linux 中设计的,用于使用标准 GNU-Linux/C 库“getopt.h”获取命令行参数。并且该库不适用于在 Windows 中构建 C 代码。
我想忽略我的代码现在所做的事情并提出以下问题。对于那些熟悉这个 C 库“getopt.h”的人来说:如果我的代码依赖于 POSIX 风格的命令行参数,是否可以在 Windows 中构建和运行我的代码? 或者我是否必须重新编写适用于 Windows 的代码,以不同的方式传递输入文件(并放弃“getopt.h”依赖项)?
getopt()
实际上是一个非常简单的功能。 我为它做了一个 github gist,代码来自这里也在下面
#include <string.h>
#include <stdio.h>
int opterr = 1, /* if error message should be printed */
optind = 1, /* index into parent argv vector */
optopt, /* character checked for validity */
optreset; /* reset getopt */
char *optarg; /* argument associated with option */
#define BADCH (int)'?'
#define BADARG (int)':'
#define EMSG ""
/*
* getopt --
* Parse argc/argv argument vector.
*/
int
getopt(int nargc, char * const nargv[], const char *ostr)
{
static char *place = EMSG; /* option letter processing */
const char *oli; /* option letter list index */
if (optreset || !*place) { /* update scanning pointer */
optreset = 0;
if (optind >= nargc || *(place = nargv[optind]) != '-') {
place = EMSG;
return (-1);
}
if (place[1] && *++place == '-') { /* found "--" */
++optind;
place = EMSG;
return (-1);
}
} /* option letter okay? */
if ((optopt = (int)*place++) == (int)':' ||
!(oli = strchr(ostr, optopt))) {
/*
* if the user didn't specify '-' as an option,
* assume it means -1.
*/
if (optopt == (int)'-')
return (-1);
if (!*place)
++optind;
if (opterr && *ostr != ':')
(void)printf("illegal option -- %c\n", optopt);
return (BADCH);
}
if (*++oli != ':') { /* don't need argument */
optarg = NULL;
if (!*place)
++optind;
}
else { /* need an argument */
if (*place) /* no white space */
optarg = place;
else if (nargc <= ++optind) { /* no arg */
place = EMSG;
if (*ostr == ':')
return (BADARG);
if (opterr)
(void)printf("option requires an argument -- %c\n", optopt);
return (BADCH);
}
else /* white space */
optarg = nargv[optind];
place = EMSG;
++optind;
}
return (optopt); /* dump back option letter */
}
你是对的。
getopt()
是POSIX,不是Windows,你通常需要重新编写所有命令行参数解析代码。
幸运的是,有一个项目,Xgetopt,适用于 Windows/MFC 类。
http://www.codeproject.com/Articles/1940/XGetopt-A-Unix-兼容-getopt-for-MFC-and-Win32
如果您可以在您的项目中实现此功能,它应该可以为您节省大量编码,并避免您必须重新进行所有解析。
此外,它还附带了一个不错的 GUI 演示应用程序,您应该会发现它很有帮助。
祝你好运!
可以使用 MinGW 运行时的代码(由 Todd C. Miller 编写):
http://sourceforge.net/apps/trac/mingw-w64/browser/trunk/mingw-w64-crt/misc
我用这些文件和CMake脚本创建了一个小库(可以生成VS项目):
如果你只是想让 getopt 在 Visual C++ 中使用而没有其他依赖,我已经从最新的 gnu libc 2.12 移植了 getopt.c,具有所有新功能。唯一的区别是你必须使用 TCHAR 而不是 char,但是这是在 Windows 中很常见。
只需下载源代码,制作,并将 libgetopt.lib 和 getopt.h getopt_int.h 复制到您的项目中即可。
您也可以使用根目录中的 CMakeList.txt 来制作它。
根据我对文档的阅读,头文件 getopt.h 特定于与 Linux(和 Hurd)一起使用的 GNU C 库。 getopt 函数本身已由 POSIX 标准化,它表示应该在 unistd.h 中与 optind optarg 等一起声明它
我自己无法在 Visual Studio 上尝试此操作,但值得检查 unistd.h 是否存在并声明此函数,因为 Visual Studio 确实提供了一些其他 POSIX 函数。
如果没有,那么我肯定会获取 getopt 的实现,而不是重新编写参数解析以在没有它的情况下工作。 Getopt 的编写是为了使程序员的工作变得更容易,并且使具有命令行参数的程序的用户更加一致。 不过,请检查许可证。
mingw-w64项目编写的最新getopt可移植实现在:
https://sourceforge.net/p/mingw-w64/mingw-w64/ci/master/tree/mingw-w64-crt/misc/getopt.c https://sourceforge.net/p/mingw-w64/mingw-w64/ci/master/tree/mingw-w64-headers/crt/getopt.h
只需将
getopt.c
与代码的目标对象一起编译即可。我已经在 Visual Studio 2022 CMake 项目上尝试过了,效果很好。
根据我对
getopt.h
的记忆,它所做的只是提供一个方便的解析器来处理主函数中的argv:
int main(int argc, char * argv[])
{
}
Windows 控制台程序仍然有一个 main 方法,因此您可以简单地循环遍历 argv 数组并自己解析参数。例如
for ( int i = 1; i < argc; i++ )
{
if (!strcmp(argv[i], "-f"))
filename = argv[++i];
}
getopt.h 存在于 git 中,我已经下载了它并且它对我有用: