文章目录

  • 1 简介
  • 2 参数格式定义
  • 3 类功能介绍
  • 3.1 初始化
  • 3.2 主要函数
  • 4 源代码
  • 5 输出结果
  • 6 小结


1 简介

在前文中,介绍了一个Python中的 argparse.ArgumentParser 类,能够实现对命令行输入参数的控制。搜索了一下,发现在Java中并没有提供相应的类。因此,为了在Java实现类似的功能,本文编写了一种用于处理main函数中的输入参考 String[] args 的类 ArgumentsParser

2 参数格式定义

首先,对参数进行定义。一般而言,命令行中的输入参数通用由多个由空格隔开,带有横杠(-)、键值(key)和值域(value)组成,每个标签格式如下所示:
-key str1 str2,比如 -path ./src ./data 多个标签间也使用空格隔开,以下是一条Java的编码示例:
javac -encoding utf8 -t -d ./bin -cp /Users/hao/servlet-api.jar ./src/*.java 其中就包括了三组标签:

  1. -encoding utf8
  2. -t
  3. -d ./bin
  4. -cp /Users/hao/servlet-api.jar ./src/*.java

其中标签2的参数个数为0,标签1,3为1,标签4的参数个数为4。

3 类功能介绍

3.1 初始化

本类使用 String[] args 作为初始化化函数:
ArgumentsParser ap = new ArgumentsParser(args); 其中 args 就是 main 函数传递进来的所有参数。

3.2 主要函数

类中的主要非静态函数的作用如下:

函数名称

作用

boolean hasKey(String key)

判断key是否存在。

Set getKeys()

返回所有的Key。

String getString(String key)

返回指定key的字符串值,如果有多个值,只返回第1个值。

String getString(String key, String defaultValue)

返回指定key的字符串值(只返回第1个值),如果不存在则返回缺省值 defaultValue。

String[] getStrings(String key)

返回指定key的所有值列表。

int getInt(String key)

返回指定key的对应的整型,如果转换失败会抛出转换失败异常。

4 源代码

import java.util.HashMap;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Set;

public class ArgumentsParser {

	public static void main(String args[]) {
		// for test
		args = "-encoding utf8 -t -print 5 -d ./bin -cp /Users/hao/servlet-api.jar ./src/*.java".split(" ");

		ArgumentsParser ap = new ArgumentsParser(args);
		System.out.println("---------- Basic Test ---------");
		String name = ap.getString("encoding");
		int prints = Integer.parseInt(ap.getString("print"));
		for (String path : ap.getStrings("paths"))
			System.out.println(path);

		String out_path = ap.hasKey("out") ? ap.getString("out") : "c:\\output";
		System.out.println("Output Path: " + out_path);

		System.out.println("---------- Get Argument Strings ---------");
		for (String name1 : ap.getKeys())
			System.out.println(name1 + ": " + ap.getStrings(name1).length);

		System.out.println("---------- Key Existance Check ---------");
		System.out.println("hasKey(\"encoding\"): " + ap.hasKey("test"));
		System.out.println("hasKey(\"name\"): " + ap.hasKey("name"));
		System.out.println("hasKey(\"cp\"): " + ap.hasKey("path"));
		System.out.println("hasKey(\"d\"): " + ap.hasKey("paths"));

		System.out.println("---------- use arguments ---------");
		for (int i = 0; i < prints; i++) {
			System.out.print(name + ", ");
		}
	}

	HashMap<String, List<String>> map = new HashMap<>();

	ArgumentsParser(String arguments[]) {
		List<String> args = Arrays.asList(arguments);
		for (String arg : args) {
			// case 1: non-hyphen started arguments.
			if (!arg.startsWith("-"))
				continue;

			// case 2: tags with 0 argument values.
			String key = arg.substring(1);
			int arg_index = args.indexOf(arg);
			if (arg_index == args.size() - 1 || args.get(arg_index + 1).startsWith("-")) {
				map.put(key, new ArrayList<String>());
				continue;
			}

			// case 3: key with 1 or more argument values
			List<String> argstrs = new ArrayList<>();
			int i = 1;
			while (arg_index + i != args.size() && !args.get(arg_index + i).startsWith("-")) {
				argstrs.add(args.get(arg_index + i));
				i++;
			}
			map.put(arg.replace("-", ""), argstrs);
		}
	}

	public boolean hasKey(String keyName) {
		if (map.containsKey(keyName))
			return true;
		return false;
	}

	// Return argument names
	public Set<String> getKeys() {
		return map.keySet();
	}

	public String getString(String keyName) {
		return getString(keyName, null);
	}

	public String getString(String keyName, String defaultValue) {
		return this.hasKey(keyName) ? map.get(keyName).get(0) : defaultValue;
	}

	public String[] getStrings(String argumentName) {
		if (map.containsKey(argumentName))
			return map.get(argumentName).toArray(new String[0]);
		else
			return new String[0];
	}

	public int getInt(String keyName, int defaultValue) {
		int retValue = 0;
		try {
			if (hasKey(keyName))
				retValue = Integer.parseInt(map.get(keyName).get(0));
		} catch (Exception ex) {
			retValue = defaultValue;
		}
		return retValue;
	}
}

5 输出结果

---------- Basic Test ---------
Output Path: c:\output
---------- Get Argument Strings ---------
print: 1
t: 0
d: 1
encoding: 1
cp: 2
---------- Key Existance Check ---------
hasKey("encoding"): false
hasKey("name"): false
hasKey("cp"): false
hasKey("d"): false
---------- use arguments ---------
utf8, utf8, utf8, utf8, utf8,

6 小结

通过本类的定义,可以方便地对命令行输入的参数进行提取,从而为后续的参数处理带来便利。