Skip to content

24.6 getopt()、getopt_long()、getopt_long_only()是何用法

Q: 能给一个getopt()的例子吗

A: 小四 scz@nsfocus.com 1999-04-05 17:02


/ * For x86/Linux * gcc -Wall -pipe -O3 -s -o getopt_test getopt_test.c /

include

include

include

include

include

define VERSION "1.00 1999-04-05 17:02"

static void usage ( char arg ) { fprintf ( stderr, "Usage: %s [-h] [-v] [-i intvar] [-s strvar]\n", arg ); exit( EXIT_FAILURE ); } / end of usage */

int main ( int argc, char * argv[] ) { int ret = EXIT_FAILURE; int c; unsigned int intvar = 0; char *strvar = NULL;

if ( 1 == argc )
{
    usage( argv[0] );
}
/*
 * don't want getopt() writing to stderr
 */
opterr  = 0;
/*
 * getopt()第三形参最前面如果有冒号,就不会出现illegal option错误信息。
 * 不过,如果指定了"opterr = 0",有无这个前导冒号都无所谓了。
 */
while ( EOF != ( c = getopt( argc, argv, ":hi:s:v" ) ) )
{
    switch ( c )
    {
    case 'i':
        intvar  = ( unsigned int )strtoul( optarg, NULL, 0 );
        break;
    case 's':
        strvar  = optarg;
        break;
    case 'v':
        printf( "%s ver "VERSION"\n", argv[0] );
        return( EXIT_SUCCESS );
    case 'h':
    case '?':
    default :
        usage( argv[0] );
        break;
    }  /* end of switch */
}  /* end of while */
argc   -= optind;
argv   += optind;
printf
(
    "intvar  = %u\n"
    "strvar  = %s\n",
    intvar,
    NULL == strvar ? "(null)" : strvar
);
ret     = EXIT_SUCCESS;
return( ret );

} / end of main /

if 0

$ ./getopt_test -h Usage: ./getopt_test [-h] [-v] [-i intvar] [-s strvar] $ ./getopt_test -i 1314 -s scz intvar = 1314 strvar = scz $ ./getopt_test -v ./getopt_test ver 1.00 1999-04-05 17:02

endif


Q: 现在会用getopt()了,但如何支持"--parami intvar --params strvar"呢

A: 小四 scz@nsfocus.com 2001-12-17 15:30

这需要用到getopt_long(),但这不是一个具有良好可移植性的函数。


/ * For x86/Linux * gcc -Wall -pipe -O3 -s -o getopt_long_test getopt_long_test.c /

include

include

include

include

include

/ * 为了使用getopt_long(),必须有如下两行内容 /

define _GNU_SOURCE

include

define VERSION "1.00 2001-12-17 15:30"

/ * 不要与短选项所用的字符相同! /

define LONGOPTIONCHAR '*'

static void usage ( char arg ) { fprintf ( stderr, "Usage: %s [-h] [--ver] [--parami intvar] [-params strvar]\n", arg ); exit( EXIT_FAILURE ); } / end of usage */

int main ( int argc, char * argv[] ) { int ret = EXIT_FAILURE; int c; unsigned int intvar = 0; char *strvar = NULL; struct option longOption[] = { { "ver", 0, NULL, LONGOPTIONCHAR }, { "parami", 1, NULL, LONGOPTIONCHAR }, { "params", 1, NULL, LONGOPTIONCHAR }, { NULL, 0, NULL, 0 } }; int longOptionIndex = 0;

if ( 1 == argc )
{
    usage( argv[0] );
}
/*
 * don't want getopt() writing to stderr
 */
opterr  = 0;
while ( EOF != ( c = getopt_long( argc, argv, ":h", longOption, &longOptionIndex ) ) )
{
    switch ( c )
    {
    case LONGOPTIONCHAR:
        if ( optarg )
        {
            switch ( longOptionIndex )
            {
            case 1:
                intvar  = ( unsigned int )strtoul( optarg, NULL, 0 );
                break;
            case 2:
                strvar  = optarg;
                break;
            default:
                break;
            }  /* end of switch */
        }
        else
        {
            switch ( longOptionIndex )
            {
            /*
             * longOption[]的下标
             */
            case 0:
                printf( "%s ver "VERSION"\n", argv[0] );
                return( EXIT_SUCCESS );
            default:
                break;
            }  /* end of switch */
        }
        break;
    case 'h':
    case '?':
    default :
        usage( argv[0] );
        break;
    }  /* end of switch */
}  /* end of while */
argc   -= optind;
argv   += optind;
printf
(
    "intvar  = %u\n"
    "strvar  = %s\n",
    intvar,
    NULL == strvar ? "(null)" : strvar
);
ret     = EXIT_SUCCESS;
return( ret );

} / end of main /

if 0

$ ./getopt_long_test -h Usage: ./getopt_long_test [-h] [--ver] [--parami intvar] [-params strvar] $ ./getopt_long_test --parami 1314 --params scz intvar = 1314 strvar = scz $ ./getopt_long_test --ver ./getopt_long_test ver 1.00 2001-12-17 15:30

endif


Q: 那getopt_long_only()与getopt_long()的区别是什么

A: 小四 scz@nsfocus.com 2005-05-09 10:48

getopt_long_only()的行为与getopt_long()非常类似,但不限于用'--'引入长选项, '-'亦可引入长选项。当'-'引入的选项不匹配任何长选项时,继续进行短选项匹配。

再次提醒,getopt_long_only()与getopt_long()一样,不具有良好可移植性。


/ * For x86/Linux * gcc -Wall -pipe -O3 -s -o getopt_long_only_test getopt_long_only_test.c /

include

include

include

include

include

/ * 为了使用getopt_long_only(),必须有如下两行内容 /

define _GNU_SOURCE

include

define VERSION "1.00 2005-05-09 10:48"

/ * 不要与短选项所用的字符相同!这次用空格符,更可靠些。 /

define LONGOPTIONCHAR ' '

static void usage ( char arg ) { fprintf ( stderr, "Usage: %s [-h] [--v] [-i intvar] [--s strvar]\n", arg ); exit( EXIT_FAILURE ); } / end of usage */

int main ( int argc, char * argv[] ) { int ret = EXIT_FAILURE; int c; unsigned int intvar = 0; char *strvar = NULL; struct option longOption[] = { { "v", 0, NULL, LONGOPTIONCHAR }, { "s", 1, NULL, LONGOPTIONCHAR }, { NULL, 0, NULL, 0 } }; int longOptionIndex = 0;

if ( 1 == argc )
{
    usage( argv[0] );
}
/*
 * don't want getopt() writing to stderr
 */
opterr  = 0;
while ( EOF != ( c = getopt_long_only( argc, argv, ":hi:", longOption, &longOptionIndex ) ) )
{
    switch ( c )
    {
    case LONGOPTIONCHAR:
        if ( optarg )
        {
            switch ( longOptionIndex )
            {
            case 1:
                strvar  = optarg;
                break;
            default:
                break;
            }  /* end of switch */
        }
        else
        {
            switch ( longOptionIndex )
            {
            /*
             * longOption[]的下标
             */
            case 0:
                printf( "%s ver "VERSION"\n", argv[0] );
                return( EXIT_SUCCESS );
            default:
                break;
            }  /* end of switch */
        }
        break;
    case 'i':
        intvar  = ( unsigned int )strtoul( optarg, NULL, 0 );
        break;
    case 'h':
    case '?':
    default :
        usage( argv[0] );
        break;
    }  /* end of switch */
}  /* end of while */
argc   -= optind;
argv   += optind;
printf
(
    "intvar  = %u\n"
    "strvar  = %s\n",
    intvar,
    NULL == strvar ? "(null)" : strvar
);
ret     = EXIT_SUCCESS;
return( ret );

} / end of main /

if 0

$ ./getopt_long_only_test -h Usage: ./getopt_long_only_test [-h] [--v] [-i intvar] [--s strvar] $ ./getopt_long_only_test --v ./getopt_long_only_test ver 1.00 2005-05-09 10:48 $ ./getopt_long_only_test -v ./getopt_long_only_test ver 1.00 2005-05-09 10:48 $ ./getopt_long_only_test -i 1314 -s scz intvar = 1314 strvar = scz

endif