Я просто зібрав приємну структуру даних і ланцюжок обробки, щоб створити цю поведінку комутації, не потрібні бібліотеки. Я впевнений, що він буде впроваджений неодноразово, і натрапив на цю тему, шукаючи приклади - подумав, що я заграю.
Мені навіть особливо не потрібні прапори (єдиний прапор тут - режим налагодження, створюючи змінну, яку я перевіряю як умову запуску функції вниз за течією if (!exists(debug.mode)) {...} else {print(variables)})
. lapply
Виписки, що перевіряють прапор нижче, дають те саме, що:
if ("--debug" %in% args) debug.mode <- T
if ("-h" %in% args || "--help" %in% args)
де args
змінна, зчитувана з аргументів командного рядка (символьний вектор, еквівалентний, c('--debug','--help')
коли, наприклад, ви їх надаєте)
Він може використовуватись повторно для будь-якого іншого прапора, і ви уникаєте всіх повторень, і жодна бібліотека, так що не залежність:
args <- commandArgs(TRUE)
flag.details <- list(
"debug" = list(
def = "Print variables rather than executing function XYZ...",
flag = "--debug",
output = "debug.mode <- T"),
"help" = list(
def = "Display flag definitions",
flag = c("-h","--help"),
output = "cat(help.prompt)") )
flag.conditions <- lapply(flag.details, function(x) {
paste0(paste0('"',x$flag,'"'), sep = " %in% args", collapse = " || ")
})
flag.truth.table <- unlist(lapply(flag.conditions, function(x) {
if (eval(parse(text = x))) {
return(T)
} else return(F)
}))
help.prompts <- lapply(names(flag.truth.table), function(x){
# joins 2-space-separatated flags with a tab-space to the flag description
paste0(c(paste0(flag.details[x][[1]][['flag']], collapse=" "),
flag.details[x][[1]][['def']]), collapse="\t")
} )
help.prompt <- paste(c(unlist(help.prompts),''),collapse="\n\n")
# The following lines handle the flags, running the corresponding 'output' entry in flag.details for any supplied
flag.output <- unlist(lapply(names(flag.truth.table), function(x){
if (flag.truth.table[x]) return(flag.details[x][[1]][['output']])
}))
eval(parse(text = flag.output))
Зауважте, що flag.details
тут команди зберігаються як рядки, а потім оцінюються за допомогою eval(parse(text = '...'))
. Optparse, очевидно, бажаний для будь-якого серйозного сценарію, але код з мінімальною функціональністю теж іноді хороший.
Вибірка зразка:
$ Rscript check_mail.Rscript --help
--debug Друк змінних, а не виконання функції XYZ ...
-h - допомогти Відображення визначень прапора