web-dev-qa-db-fra.com

Passer des arguments nommés aux scripts Shell

Existe-t-il un moyen simple de passer (recevoir) des paramètres nommés à un script Shell?

Par exemple,

my_script -p_out '/some/path' -arg_1 '5'

Et à l'intérieur my_script.sh les recevoir comme:

# I believe this notation does not work, but is there anything close to it?
p_out=$ARGUMENTS['p_out']
arg1=$ARGUMENTS['arg_1']

printf "The Argument p_out is %s" "$p_out"
printf "The Argument arg_1 is %s" "$arg1"

Est-ce possible dans Bash ou Zsh?

134

Si cela ne vous dérange pas d'être limité aux noms d'arguments à une seule lettre, c'est-à-dire my_script -p '/some/path' -a5, puis en bash, vous pouvez utiliser le getopts intégré, par exemple.

#!/bin/bash

while getopts ":a:p:" opt; do
  case $opt in
    a) arg_1="$OPTARG"
    ;;
    p) p_out="$OPTARG"
    ;;
    \?) echo "Invalid option -$OPTARG" >&2
    ;;
  esac
done

printf "Argument p_out is %s\n" "$p_out"
printf "Argument arg_1 is %s\n" "$arg_1"

Ensuite, vous pouvez faire

$ ./my_script -p '/some/path' -a5
Argument p_out is /some/path
Argument arg_1 is 5

Il y a un utile Petit tutoriel getopts ou vous pouvez taper help getopts à l'invite du shell.

170
steeldriver

J'ai volé ça à drupal.org , mais vous pourriez faire quelque chose comme ça:

while [ $# -gt 0 ]; do
  case "$1" in
    --p_out=*)
      p_out="${1#*=}"
      ;;
    --arg_1=*)
      arg_1="${1#*=}"
      ;;
    *)
      printf "***************************\n"
      printf "* Error: Invalid argument.*\n"
      printf "***************************\n"
      exit 1
  esac
  shift
done

La seule mise en garde est que vous devez utiliser la syntaxe my_script --p_out=/some/path --arg_1=5.

43
cdmo

J'utilise ce script et fonctionne comme un charme:

for ARGUMENT in "$@"
do

    KEY=$(echo $ARGUMENT | cut -f1 -d=)
    VALUE=$(echo $ARGUMENT | cut -f2 -d=)   

    case "$KEY" in
            STEPS)              STEPS=${VALUE} ;;
            REPOSITORY_NAME)    REPOSITORY_NAME=${VALUE} ;;     
            *)   
    esac    


done

echo "STEPS = $STEPS"
echo "REPOSITORY_NAME = $REPOSITORY_NAME"

Usage

bash my_scripts.sh  STEPS="ABC" REPOSITORY_NAME="stackexchange"

Résultat de la console:

STEPS = ABC
REPOSITORY_NAME = stackexchange

ÉTAPES et REPOSITORY_NAME sont prêts à l'emploi dans le script.

Peu importe l'ordre dans lequel les arguments sont.

36
JRichardsz

La syntaxe probablement la plus proche de cela est:

p_out='/some/path' arg_1='5' my_script
36
Hauke Laging

Avec zsh, vous utiliseriez zparseopts:

#! /bin/zsh -
zmodload zsh/zutil
zparseopts -A ARGUMENTS -p_out: -arg_1:

p_out=$ARGUMENTS[--p_out]
arg1=$ARGUMENTS[--arg_1]

printf 'Argument p_out is "%s"\n' "$p_out"
printf 'Argument arg_1 is "%s"\n' "$arg_1"

Mais vous appelleriez le script avec myscript --p_out foo.

Notez que zparseopts ne prend pas en charge l'abréviation des options longues ou la syntaxe --p_out=foo Comme GNU getopt(3) le fait.

15

Je viens de créer ce script

while [ $# -gt 0 ]; do

   if [[ $1 == *"--"* ]]; then
        v="${1/--/}"
        declare $v="$2"
   fi

  shift
done

passez-le comme my_script --p_out /some/path --arg_1 5 puis dans le script, vous pouvez utiliser $arg_1 et $p_out.

12
Shahzad Malik