如何在Java中解析命令行参数?

在Java中解析命令行参数的好方法是什么?


#1楼

我不建议使用Apache Common CLI库,因为它是非线程安全的。 它使用带有静态变量和方法的有状态类来进行内部工作(例如OptionBuilder),并且仅应在单线程严格控制的情况下使用。


#2楼

我写了另一个: http : //argparse4j.sourceforge.net/

Argparse4j是基于Python的argparse的Java命令行参数解析器库。


#3楼

@ Github的航空公司看起来不错。 它基于注释,正在尝试模拟Git命令行结构。


#4楼

看一下最近的JCommander

我创造了它。 我很高兴收到问题或功能要求。


#5楼

检查这些:

或自己动手:


例如,这是使用commons-cli解析2个字符串参数的方式:

import org.apache.commons.cli.*;

public class Main {


    public static void main(String[] args) throws Exception {

        Options options = new Options();

        Option input = new Option("i", "input", true, "input file path");
        input.setRequired(true);
        options.addOption(input);

        Option output = new Option("o", "output", true, "output file");
        output.setRequired(true);
        options.addOption(output);

        CommandLineParser parser = new DefaultParser();
        HelpFormatter formatter = new HelpFormatter();
        CommandLine cmd;

        try {
            cmd = parser.parse(options, args);
        } catch (ParseException e) {
            System.out.println(e.getMessage());
            formatter.printHelp("utility-name", options);

            System.exit(1);
        }

        String inputFilePath = cmd.getOptionValue("input");
        String outputFilePath = cmd.getOptionValue("output");

        System.out.println(inputFilePath);
        System.out.println(outputFilePath);

    }

}

从命令行使用:

$> java -jar target/my-utility.jar -i asd                                                                                       
Missing required option: o

usage: utility-name
 -i,--input <arg>    input file path
 -o,--output <arg>   output file

#6楼

看一下Commons CLI项目,那里有很多不错的东西。


#7楼

也许这些


#8楼

我使用过JOpt并发现它非常方便: http ://jopt-simple.sourceforge.net/

主页还提供了大约8个替代库的列表,将其检出并选择最适合您需要的一个。


#9楼

是的

我认为您正在寻找这样的东西: http : //commons.apache.org/cli

Apache Commons CLI库提供了用于处理命令行界面的API。


#10楼

如果您熟悉gnu getopt,则在以下网址有一个Java端口: http : //www.urbanophile.com/arenn/hacking/download.htm

似乎有一些类可以做到这一点:


#11楼

您可能会发现这不愉快的元文章很有趣,可以作为起点:

http://furiouspurpose.blogspot.com/2008/07/command-line-parsing-libraries-for-java.html


#12楼

最近有人将我指向基于注释的args4j 。 我很喜欢!


#13楼

这是Google命令行解析库,是Bazel项目的一部分,是开源的。 我个人认为这是目前最好的解决方案,并且比Apache CLI容易得多。

https://github.com/pcj/google-options

安装

淡褐色

maven_jar(
    name = "com_github_pcj_google_options",
    artifact = "com.github.pcj:google-options:jar:1.0.0",
    sha1 = "85d54fe6771e5ff0d54827b0a3315c3e12fdd0c7",
)

摇篮

dependencies {
  compile 'com.github.pcj:google-options:1.0.0'
}

马文

<dependency>
  <groupId>com.github.pcj</groupId>
  <artifactId>google-options</artifactId>
  <version>1.0.0</version>
</dependency>

用法

创建一个扩展OptionsBase的类并定义您的@Option

package example;

import com.google.devtools.common.options.Option;
import com.google.devtools.common.options.OptionsBase;

import java.util.List;

/**
 * Command-line options definition for example server.
 */
public class ServerOptions extends OptionsBase {

  @Option(
      name = "help",
      abbrev = 'h',
      help = "Prints usage info.",
      defaultValue = "true"
    )
  public boolean help;

  @Option(
      name = "host",
      abbrev = 'o',
      help = "The server host.",
      category = "startup",
      defaultValue = ""
  )
  public String host;

  @Option(
    name = "port",
    abbrev = 'p',
    help = "The server port.",
    category = "startup",
    defaultValue = "8080"
    )
    public int port;

  @Option(
    name = "dir",
    abbrev = 'd',
    help = "Name of directory to serve static files.",
    category = "startup",
    allowMultiple = true,
    defaultValue = ""
    )
    public List<String> dirs;

}

解析参数并使用它们。

package example;

import com.google.devtools.common.options.OptionsParser;
import java.util.Collections;

public class Server {

  public static void main(String[] args) {
    OptionsParser parser = OptionsParser.newOptionsParser(ServerOptions.class);
    parser.parseAndExitUponError(args);
    ServerOptions options = parser.getOptions(ServerOptions.class);
    if (options.host.isEmpty() || options.port < 0 || options.dirs.isEmpty()) {
      printUsage(parser);
      return;
    }

    System.out.format("Starting server at %s:%d...\n", options.host, options.port);
    for (String dirname : options.dirs) {
      System.out.format("\\--> Serving static files at <%s>\n", dirname);
    }
  }

  private static void printUsage(OptionsParser parser) {
    System.out.println("Usage: java -jar server.jar OPTIONS");
    System.out.println(parser.describeOptions(Collections.<String, String>emptyMap(),
                                              OptionsParser.HelpVerbosity.LONG));
  }

}

https://github.com/pcj/google-options


#14楼

现在是2019年,是时候比Commons CLI做得更好了... :-)

购买还是建造?

许多类似小型实用程序的应用程序可能会滚动其自己的命令行解析,以避免额外的外部依赖性。

picocli可能很有趣。 它被设计为作为源提供,作为将jar着色到ub​​erjar中的一种更简单的替代方法。

您可能喜欢的另一个功能是其彩色使用帮助。

最少使用ANSI颜色帮助

解析器功能:

  • 基于注释:解析是一行代码
  • 严格键入所有内容-命令行选项以及位置参数
  • POSIX集群短选项( <command> -xvfInputFile以及<command> -x -v -f InputFile
  • 允许最小,最大和可变数量参数的Arity模型,例如"1..*""3..5"
  • 子命令(可以嵌套到任意深度)
  • 适用于Java 5及更高版本

使用帮助消息很容易通过注释进行自定义(无需编程)。 例如:

扩展使用帮助信息来源

我忍不住要再添加一个屏幕截图,以显示可能的使用帮助消息。 使用帮助是您应用程序的面目,因此请发挥创造力并尽情玩乐!

picocli演示

免责声明:我创建了picocli。 反馈或问题非常欢迎。


#15楼

如果您想要轻量级的东西(罐子大小〜20 kb)并且易于使用,则可以尝试arguments-parser 。 它可以在大多数用例中使用,支持在参数中指定数组,并且不依赖于任何其他库。 它适用于Java 1.5或更高版本。 下面的摘录显示了有关如何使用它的示例:

public static void main(String[] args) {
    String usage = "--day|-d day --mon|-m month [--year|-y year][--dir|-ds directoriesToSearch]";
    ArgumentParser argParser = new ArgumentParser(usage, InputData.class);
    InputData inputData = (InputData) argParser.parse(args);
    showData(inputData);

    new StatsGenerator().generateStats(inputData);
}

可以在这里找到更多示例


#16楼

我发现Argparse4j是最好的。 它模仿Python的argparse库,该库非常方便且强大。


#17楼

我知道这里的大多数人会发现一千万个为什么他们不喜欢我的方式但没关系的原因。 我喜欢使事情保持简单,因此我只是使用'='将键与值分开,并将它们存储在HashMap中,如下所示:

Map<String, String> argsMap = new HashMap<>();
for (String arg: args) {
    String[] parts = arg.split("=");
    argsMap.put(parts[0], parts[1]);
} 

您始终可以维护一个包含期望参数的列表,以在用户忘记参数或使用错误参数时为用户提供帮助。但是,如果您想要太多功能,则无论如何该解决方案都不适合您。


#18楼

正如前面提到的评论之一( https://github.com/pcj/google-options )是一个不错的选择。

我想添加的一件事是:

1)如果遇到解析器反射错误,请尝试使用较新版本的番石榴。 就我而言:

maven_jar(
    name = "com_google_guava_guava",
    artifact = "com.google.guava:guava:19.0",
    server = "maven2_server",
)

maven_jar(
    name = "com_github_pcj_google_options",
    artifact = "com.github.pcj:google-options:jar:1.0.0",
    server = "maven2_server",
)

maven_server(
    name = "maven2_server",
    url = "http://central.maven.org/maven2/",
)

2)运行命令行时:

bazel run path/to/your:project -- --var1 something --var2 something -v something

3)当需要使用帮助时,只需键入:

bazel run path/to/your:project -- --help

#19楼

对于Spring用户,我们还应该提到https://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/core/env/SimpleCommandLinePropertySource.html和他的双胞胎兄弟https://docs.spring .io / spring / docs / current / javadoc-api / org / springframework / core / env / JOptCommandLinePropertySource.html (具有相同功能的JOpt实现)。 Spring的优势在于您可以将命令行参数直接绑定到属性,这里有一个示例https://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/core/env/ CommandLinePropertySource.html


#20楼

如果您已经在使用Spring Boot,则可以直接进行参数解析。

如果要在启动后运行某些程序,请实现ApplicationRunner接口:

@SpringBootApplication
public class Application implements ApplicationRunner {

  public static void main(String[] args) {
    SpringApplication.run(Application.class, args);
  }

  @Override
  public void run(ApplicationArguments args) {
    args.containsOption("my-flag-option"); // test if --my-flag-option was set
    args.getOptionValues("my-option");     // returns values of --my-option=value1 --my-option=value2 
    args.getOptionNames();                 // returns a list of all available options
    // do something with your args
  }
}

上下文成功启动后,将调用您的run方法。

如果启动应用程序上下文之前需要访问参数,则可以简单地手动解析应用程序参数:

@SpringBootApplication
public class Application implements ApplicationRunner {

  public static void main(String[] args) {
    ApplicationArguments arguments = new DefaultApplicationArguments(args);
    // do whatever you like with your arguments
    // see above ...
    SpringApplication.run(Application.class, args);
  }

}

最后,如果需要访问Bean中的参数,只需注入ApplicationArguments

@Component
public class MyBean {

   @Autowired
   private ApplicationArguments arguments;

   // ...
}

#21楼

我一直在尝试维护Java CLI解析器列表

发布了0 篇原创文章 · 获赞 1 · 访问量 2547

猜你喜欢

转载自blog.csdn.net/asdfgh0077/article/details/103937623