首页 程序笔记 简单、小巧、灵活的C++11单头文件的命令行参数解析库

简单、小巧、灵活的C++11单头文件的命令行参数解析库

概述

命令行参数解析,在写一些命令行的程序时需要用到,args.hxx类似于Python的argparse,但在C++中,它具有静态类型检查,并且速度更快(也允许完全嵌套的组逻辑,而Python的argparse没有),支持自定义类型解析,子命令等。使用上,只需要在工程中include “args.hxx” 就可以正常使用,符合C++11规范,兼容性更好。

文档和源码

API文档地址: https://taywee.github.io/args
源码地址: at https://github.com/Taywee/args

范例

以下的所有代码示例都将是完整的代码示例,并带有一些输出。

简单范例

#include iostream
#include args.hxx
int main(int argc, char **argv)
{
    args::ArgumentParser parser(This is a test program., This goes after the options.);
    args::HelpFlag help(parser, help, Display this help menu, {h, help});
    args::CompletionFlag completion(parser, {complete});
    try
    {
        parser.ParseCLI(argc, argv);
    }
    catch (const args::Completion e)
    {
        std::cout  e.what();
        return 0;
    }
    catch (const args::Help)
    {
        std::cout  parser;
        return 0;
    }
    catch (const args::ParseError e)
    {
        std::cerr  e.what()  std::endl;
        std::cerr  parser;
        return 1;
    }
    return 0;
}

运行输出:

 % ./test
 % ./test -h
  ./test {OPTIONS} 

    This is a test program. 

  OPTIONS:

      -h, --help         Display this help menu 

    This goes after the options. 
 % 

布尔标志、特殊组类型、不同的匹配器构造

#include iostream
#include args.hxx
int main(int argc, char **argv)
{
    args::ArgumentParser parser(This is a test program., This goes after the options.);
    args::Group group(parser, This group is all exclusive:, args::Group::Validators::Xor);
    args::Flag foo(group, foo, The foo flag, {f, foo});
    args::Flag bar(group, bar, The bar flag, {b});
    args::Flag baz(group, baz, The baz flag, {baz});
    try
    {
        parser.ParseCLI(argc, argv);
    }
    catch (args::Help)
    {
        std::cout  parser;
        return 0;
    }
    catch (args::ParseError e)
    {
        std::cerr  e.what()  std::endl;
        std::cerr  parser;
        return 1;
    }
    catch (args::ValidationError e)
    {
        std::cerr  e.what()  std::endl;
        std::cerr  parser;
        return 1;
    }
    if (foo) { std::cout  foo  std::endl; }
    if (bar) { std::cout  bar  std::endl; }
    if (baz) { std::cout  baz  std::endl; }
    return 0;
}

运行输出:

 % ./test   
Group validation failed somewhere!
  ./test {OPTIONS} 

    This is a test program. 

  OPTIONS:

                         This group is all exclusive:
        -f, --foo          The foo flag 
        -b                 The bar flag 
        --baz              The baz flag 

    This goes after the options. 
 % ./test -f
foo
 % ./test --foo
foo
 % ./test --foo -f
foo
 % ./test -b      
bar
 % ./test --baz
baz
 % ./test --baz -f
Group validation failed somewhere!
  ./test {OPTIONS} 

    This is a test program. 
...
 % ./test --baz -fb
Group validation failed somewhere!
  ./test {OPTIONS} 
...
 % 

参数标志、位置参数、列表

#include iostream
#include args.hxx
int main(int argc, char **argv)
{
    args::ArgumentParser parser(This is a test program., This goes after the options.);
    args::HelpFlag help(parser, help, Display this help menu, {h, help});
    args::ValueFlagint integer(parser, integer, The integer flag, {i});
    args::ValueFlagListchar characters(parser, characters, The character flag, {c});
    args::Positionalstd::string foo(parser, foo, The foo position);
    args::PositionalListdouble numbers(parser, numbers, The numbers position list);
    try
    {
        parser.ParseCLI(argc, argv);
    }
    catch (args::Help)
    {
        std::cout  parser;
        return 0;
    }
    catch (args::ParseError e)
    {
        std::cerr  e.what()  std::endl;
        std::cerr  parser;
        return 1;
    }
    catch (args::ValidationError e)
    {
        std::cerr  e.what()  std::endl;
        std::cerr  parser;
        return 1;
    }
    if (integer) { std::cout  i:   args::get(integer)  std::endl; }
    if (characters) { for (const auto ch: args::get(characters)) { std::cout  c:   ch  std::endl; } }
    if (foo) { std::cout  f:   args::get(foo)  std::endl; }
    if (numbers) { for (const auto nm: args::get(numbers)) { std::cout  n:   nm  std::endl; } }
    return 0;
}

运行输出:

% ./test -h
  ./test {OPTIONS} [foo] [numbers...] 

    This is a test program. 

  OPTIONS:

      -h, --help         Display this help menu 
      -i integer         The integer flag 
      -c characters      The character flag 
      foo                The foo position 
      numbers            The numbers position list 
      -- can be used to terminate flag options and force all following
      arguments to be treated as positional options 

    This goes after the options. 
 % ./test -i 5
i: 5
 % ./test -i 5.2
Argument integer received invalid value type 5.2
  ./test {OPTIONS} [foo] [numbers...] 
 % ./test -c 1 -c 2 -c 3
c: 1
c: 2
c: 3
 % 
 % ./test 1 2 3 4 5 6 7 8 9
f: 1
n: 2
n: 3
n: 4
n: 5
n: 6
n: 7
n: 8
n: 9
 % ./test 1 2 3 4 5 6 7 8 9 a
Argument numbers received invalid value type a
  ./test {OPTIONS} [foo] [numbers...] 

    This is a test program. 
...

命令行子命令

#include iostream
#include args.hxx
int main(int argc, char **argv)
{
    args::ArgumentParser p(git-like parser);
    args::Group commands(p, commands);
    args::Command add(commands, add, add file contents to the index);
    args::Command commit(commands, commit, record changes to the repository);
    args::Group arguments(p, arguments, args::Group::Validators::DontCare, args::Options::Global);
    args::ValueFlagstd::string gitdir(arguments, path, , {git-dir});
    args::HelpFlag h(arguments, help, help, {h, help});
    args::PositionalListstd::string pathsList(arguments, paths, files to commit);

    try
    {
        p.ParseCLI(argc, argv);
        if (add)
        {
            std::cout  Add;
        }
        else
        {
            std::cout  Commit;
        }

        for (auto path : pathsList)
        {
            std::cout     path;
        }

        std::cout  std::endl;
    }
    catch (args::Help)
    {
        std::cout  p;
    }
    catch (args::Error e)
    {
        std::cerr  e.what()  std::endl  p;
        return 1;
    }
    return 0;
}

运行输出:

% ./test -h
  ./test COMMAND [paths...] {OPTIONS}

    git-like parser

  OPTIONS:

      commands
        add                               add file contents to the index
        commit                            record changes to the repository
      arguments
        --git-dir=[path]
        -h, --help                        help
        paths...                          files
      -- can be used to terminate flag options and force all following
      arguments to be treated as positional options

% ./test add 1 2
Add 1 2

重构子命令

#include iostream
#include args.hxx

args::Group arguments(arguments);
args::ValueFlagstd::string gitdir(arguments, path, , {git-dir});
args::HelpFlag h(arguments, help, help, {h, help});
args::PositionalListstd::string pathsList(arguments, paths, files to commit);

void CommitCommand(args::Subparser parser)
{
    args::ValueFlagstd::string message(parser, MESSAGE, commit message, {m});
    parser.Parse();

    std::cout  Commit;

    for (auto path : pathsList)
    {
        std::cout     path;
    }

    std::cout  std::endl;

    if (message)
    {
        std::cout  message:   args::get(message)  std::endl;
    }
}

int main(int argc, const char **argv)
{
    args::ArgumentParser p(git-like parser);
    args::Group commands(p, commands);
    args::Command add(commands, add, add file contents to the index, [](args::Subparser parser)
    {
        parser.Parse();
        std::cout  Add;

        for (auto path : pathsList)
        {
            std::cout     path;
        }

        std::cout  std::endl;
    });

    args::Command commit(commands, commit, record changes to the repository, CommitCommand);
    args::GlobalOptions globals(p, arguments);

    try
    {
        p.ParseCLI(argc, argv);
    }
    catch (args::Help)
    {
        std::cout  p;
    }
    catch (args::Error e)
    {
        std::cerr  e.what()  std::endl  p;
        return 1;
    }
    return 0;
}

运行结果:

% ./test -h
  ./test COMMAND [paths...] {OPTIONS}

    git-like parser

  OPTIONS:

      commands
        add                               add file contents to the index
        commit                            record changes to the repository
      arguments
        --git-dir=[path]
        -h, --help                        help
        paths...                          files
      -- can be used to terminate flag options and force all following
      arguments to be treated as positional options

% ./test add 1 2
Add 1 2

% ./test commit -m my commit message 1 2
Commit 1 2
message: my commit message

自定义类型解析器

这里我们使用std::tuple

#include iostream
#include tuple

std::istream operator(std::istream is, std::tupleint, int ints)
{
    is  std::get0(ints);
    is.get();
    is  std::get1(ints);
    return is;
}

#include args.hxx

struct DoublesReader
{
    void operator()(const std::string name, const std::string value, std::tupledouble, double destination)
    {
        size_t commapos = 0;
        std::get0(destination) = std::stod(value, commapos);
        std::get1(destination) = std::stod(std::string(value, commapos + 1));
    }
};

int main(int argc, char **argv)
{
    args::ArgumentParser parser(This is a test program.);
    args::Positionalstd::tupleint, int ints(parser, INTS, This takes a pair of integers.);
    args::Positionalstd::tupledouble, double, DoublesReader doubles(parser, DOUBLES, This takes a pair of doubles.);
    try
    {
        parser.ParseCLI(argc, argv);
    }
    catch (args::Help)
    {
        std::cout  parser;
        return 0;
    }
    catch (args::ParseError e)
    {
        std::cerr  e.what()  std::endl;
        std::cerr  parser;
        return 1;
    }
    if (ints)
    {
        std::cout  ints found:   std::get0(args::get(ints))   and   std::get1(args::get(ints))  std::endl;
    }
    if (doubles)
    {
        std::cout  doubles found:   std::get0(args::get(doubles))   and   std::get1(args::get(doubles))  std::endl;
    }
    return 0;
}

运行输出:

 % ./test -h
Argument could not be matched: h
  ./test [INTS] [DOUBLES] 

    This is a test program. 

  OPTIONS:

      INTS               This takes a pair of integers. 
      DOUBLES            This takes a pair of doubles. 

 % ./test 5
ints found: 5 and 0
 % ./test 5,8
ints found: 5 and 8
 % ./test 5,8 2.4,8
ints found: 5 and 8
doubles found: 2.4 and 8
 % ./test 5,8 2.4, 
terminate called after throwing an instance of std::invalid_argument
  what():  stod
zsh: abort      ./test 5,8 2.4,
 % ./test 5,8 2.4 
terminate called after throwing an instance of std::out_of_range
  what():  basic_string::basic_string: __pos (which is 4)  this-size() (which is 3)
zsh: abort      ./test 5,8 2.4
 % ./test 5,8 2.4-7
ints found: 5 and 8
doubles found: 2.4 and 7
 % ./test 5,8 2.4,-7
ints found: 5 and 8
doubles found: 2.4 and -7

更多范例请参考github上的手册:https://github.com/Taywee/args

3

站心网

概述 命令行参数解析,在写一些命令行的程序时需要用到,args.hxx类似于Python的argparse,但在C++中,它具..

为您推荐

轻松学习 JavaScript函数中的默认参数

JavaScript函数可以有默认参数值。通过默认函数参数,你可以初始化带有默认值的正式参数。如果不初始化具有某些值的参数,则该参数的默认值为undefined。请看下列代码:function foo(num1){console.log(num1);}foo()..

7大Git实用技巧和命令总结

Git 是一个非常强大的工具,它包含丰富的工具用以维护项目。本文介绍了一些 Git 日常使用过程中的实用技巧和命令,希望这些内容能够对大家有所帮助。Git diff通常情况下,我们会在自己的独立分支中完成需求开发,此..

回首2018,展望2019

今年这个春节,《流浪地球》大火,路人皆知,准确的说,连银河系都知道地球要跑路了,还顺带无情的抛弃了月亮。所以,太阳不高兴了,整个春节也没给我们什么好脸,甚至连脸都没露~人,有时候很感性。风和日丽的时候..

mysql分表简单介绍

一、Mysql分表的原因1、当一张的数据达到几百万时,你查询一次所花的时间会变多,如果有联合查询的话,我想有可能会停在那儿了。分表的目的就在于此,减小数据库的负担,缩短查询时间。2、mysql中有一种机制是表锁定..

一个简单的 C# 异步日志记录器

Clearcove.Logging是一个非常简单的日志库,旨在通过直接许可条款满足大多数日志记录需求。介绍我知道你在想什么——代码世界真的需要另一个日志库吗?如果你在.NET中寻找一个日志库,那么你有很多选择。有NLog,Log..

C#中数组作为参数传递的问题

原则:尽可能控制对数据的修改,如果可以预测某个数据不会或不应该被改变,就要对其控制,而不要期望使用这个数据的调用者不会改变其值。如果参数在使用过程中被意外修改,将会带来不可预知的结果,而且这种错误很难..

JS获取url参数

以下JS函数用于获取url参数:functiongetQueryVariable(variable){varquery=window.location.search.substring(1);varvars=query.split("&");for(vari=0;i<vars.length;i++){varpair=vars[i].split("=");if(pair[0]==..

.NET 主程序的.dll.config文件有什么用?

.dll.config 文件通常是在开发过程中自动生成的,它的主要作用是为某个类库(.dll 文件)提供一个独立的配置文件,以便开发者可以为该类库单独定义或测试配置项。以下是生成 .dll.config 文件的原因和机制:1. 配置..

一个简单的大转盘抽奖程序(附.NetCore Demo源码)

最近闲下来在做一些demo,现在讲一下做的一个简单的大转盘抽奖demo,前端lottery,layui,后端.net core,sqlsugar,数据库用的mysql1.前端实现:前端用的是基于开源的lottery,其中有些改动的,使得前端可以自适应pc端..

.NET C#连接FTP实现文件上传下载

在 .NET 中可以使用 System.Net.FtpWebRequest 类来连接 FTP 服务器,实现文件上传和下载。以下是实现文件上传和下载的完整代码示例。1. 上传文件到 FTP 服务器using System;using System.IO;using System.Net;class..

.net 通过 HttpClient 下载文件同时报告进度的方法

通过 HttpClient 的 ContentLength 很多时候都可以拿到下载的内容的长度,通过 ReadAsync 可以返回当前读到的长度,将读取到的长度加起来就是已经下载的长度看起来很简单,于是直接给代码private static async Task ..

ASP.NET如何将Views文件夹从项目分离

将 Views 文件夹从 ASP.NET 项目中分离是一个常见需求,比如为了实现模块化或分层架构。以下是实现此功能的完整步骤,从项目中分离 Views 文件夹,将其移到另一个独立的文件夹或项目中,并确保视图渲染仍然正常。1. ..

.NET C# 读取编辑.AVIF图片文件

在 .NET 中读取和编辑 .AVIF 图片文件需要特定的库支持,因为 System.Drawing 等内置功能不直接支持 AVIF 格式。目前可以通过以下方式在 .NET 中实现对 AVIF 文件的读取和编辑:方法一:使用 ImageMagick 的 .NET 封..

.NET C# SkiaSharp读取.AVIF图片文件报错

SkiaSharp 目前对 .AVIF 格式的支持可能依赖于具体的版本和底层库的配置。如果在使用 SkiaSharp 时尝试读取 .AVIF 文件报错,以下是一些可能的原因和解决方案:1. 检查 SkiaSharp 的版本SkiaSharp 的支持功能取决于..

无法加载文件或程序集 'XXXXX' 或其依赖项。访问被拒绝

遇到 “无法加载文件或程序集 'XXXXX' 或其依赖项。访问被拒绝” 错误时,通常是由于权限问题或文件夹、程序集引用配置不当所引起。下面是一些常见的原因及解决方法:1. 文件或程序集权限问题如果服务器或..

Git 常用命令总结

基础命令初始化一个Git仓库:git init添加到暂存区:git add <file>提交到本地仓库:git commit -m <message>仓库版本号切换:git reset --hard commit_id查看提交历史:git log查看历史命令:git reflog丢弃工作区..

简单优雅的Java ORM

Java的ORM框架有很多,但由于Java语言的限制大部分都不够优雅也不够简单,所以作者只能另辟蹊径造轮子了。照旧先看示例代码了解个大概,然后再解释实现原理。一、ORM示例Insertpublic CompletableFuture<Void> inser..

.NET 9 即将推出的功能Task.WhenEach

.NET 爱好者!我刚刚偶然发现了一个非常酷的新 PR,它被合并到 .NET 运行时存储库中,我想分享一个例子。希望您能为新的 .NET 版本大肆宣传!在即将到来的 .NET 9 版本中,我们预计会有一个名为 .它在这里让您的异步..

针对 Go 语言开发的 SQL 驱动模拟库

数据库交互是几乎所有应用程序不可或缺的一部分,开发者们常常需要对数据库进行各种操作,包括插入、更新、删除和查询等。然而,在开发过程中直接对真实数据库进行操作不仅耗时耗力,还可能带来数据一致性和安全性的..

文件上传JavaScript库FilePond使用教程

传统的文件上传控件往往显得笨拙且不够用户友好。FilePond的出现,为Web文件上传带来了革命性的改变。本文将详细介绍FilePond这一JavaScript库,探讨它如何优化文件上传流程,并提供无与伦比的用户体验。什么是FileP..

发表回复

返回顶部