Васильев Андрей Михайлович, 2025
Версии презентации
Официальная документация: https://jcommander.org/
JavaDoc-документация: https://javadoc.io/doc/com.beust/jcommander/latest/index.html
Альтернаитвное Java-решение: https://picocli.info/
Для работы с библиотекой, её необходимо добавить в список зависимостей приложения
Gradle-описание зависимости для подключения: "org.jcommander:jcommander:2.0"
При использовании Gradle без дополнений её необходимо добавить в файл
build.gradle.kts
в раздел dependencies
:
dependencies {
implementation("org.jcommander:jcommander:2.0")
}
При использовании шаблона лабораторной добавьте строку в файл
build.properties.json
в список dependencies
@Parameters(separators = "=")
class Args {
@Parameter
var targets: List<String> = arrayListOf()
@Parameter(names = ["-bf", "--buildFile"],
description = "The build file")
var buildFile: String? = null
@Parameter(names = ["--checkVersions"],
description = "Check if there are any newer versions")
var checkVersions = false
}
separators="="
включается обработка аргументов в
формате ключ=значение
помимо формата ключ значение
names
val commander: JCommander =
JCommander
.newBuilder()
.addObject(commonParameters)
.build()
newBuilder()
addObject()
build()
parse()
, принимающего произвольное
количество строкcommander.parse("--enable", "--size", "15")
val arguments: Array<String> = ...
commander.parse(*arguments)
commonParameters
JCommander постарается автоматически преобразовать данные из строкового представления в тип данных свойства класса для: логических полей, целых чисел, вещественных чисел и строк
Для других сложных типов данных, классов можно предоставить класс-преобразователь, который должен реализовывать следующий Java-интерфейс:
public interface IStringConverter<T> {
T convert(String value);
}
Например для UUID на Kotlin такой класс может выглядеть следующим образом:
class UUIDConverter : IStringConverter<UUID> {
override fun convert(parameter: String): UUID =
UUID.fromString(parameter)
}
В целевом объекте-хранителе аргументов необходимо сослаться на данный преобразователь следующим образом:
@Parameter(names=["--id"], converter = UUIDConverter::class)
var someId: UUID? = null
Аннотация Parameter позволяет указать аргумент как обязательный:
@Parameter(names=["--data-one"], required = true)
var dataOne: Int = 0
Согласно документации в случае отсутствия аргумента будет выброшено соответствующее исключение. Однако в данном случае оно не будет выброшено
null
, то
будет выброшено исключениеСледовательно надо обязательные аргументы описывать так:
@Parameter(names=["--data-one"], required = true)
var dataOne: Int? = null
Факт отсутствия исключения не убеждает компилятор Kotlin в наличие
не-null
-значения в свойстве объекта.
Приложения с командным интерфейсом зачастую предоставляют ряд связных между собой действий, каждое из которых имеет свой собственный набор аргументов
git clone <REPOSITORY>
git switch --create <BRANCH>
Названием команды обычно выступает первый позиционный аргумент приложения
Согласно концепции JCommander
parsedCommand
Описание не отличается от обычного объекта-хранителя:
@Parameters(separators = "=",
commandDescription = "List application entries")
class ListCommand {
@Parameter(names = ["-s", "--size"],
description = "Amount of items to display")
var size: Int = 10
}
Единственным отличием является возможность использования аргумента
commandDescription
у аннотации Parameters, которая позволяет добавить описание
назначения команды
val commander: JCommander =
JCommander
.newBuilder()
.addCommand("list", listCommand)
.addCommand("build", buildCommand)
.build()
newBuilder()
commander.parse("list", "--size")
val arguments: Array<String> = ...
commander.parse(*arguments)
if (commander.parsedCommand == "list") {
... // Обрабатываем данные внутри listCommand
} else if (commander.parsedCommand == "build") {
... // Обрабатываем данные внутри buildCommand
}
JCommander способен сформировать справку по использованию всех доступных аргументов, их значений по умолчанию и т.д.
Для вывода этой информации на стандартный поток вывода, вызовите метод usage()
на объекте JCommander после добавления к нему объектов и команд
Usage: <main class> [command] [command options]
Commands:
list List application entries
Usage: list [options]
Options:
-s, --size
Amount of items to display
Default: 10
update Update application entry
Usage: update [options]
Options:
* -i, --id
Id of the entry to update
Аргументы, помеченные *
, являются обязательными
При разработке приложения с большим количеством аргументов надо иметь возможность удобным образом переключаться между разными наборами аргументов
Помимо использования модульных тестов JCommander предоставляет возможность
считывания списка аргументов из файла с помощью @
-синтаксиса
commander.parse("@params.txt")
Из файла params.txt
, который находится в текущем рабочем каталоге:
#
, будут проигнорированы@
-аргументыДанный подход удобнее постоянного редактирования конфигурации запуска Gradle-задачи