:处理用户输入

Posted zcj仲从建

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了:处理用户输入相关的知识,希望对你有一定的参考价值。

13.1 命令行参数

  • 命令行参数:允许在运行脚本时向命令行添加数据值
$ ./addem 10 30

13.1.1 读取参数

  • bash shell会将一些称为位置参数的特殊变量分配给命令行输入的所有参数,者甚至包括shell执行的程序的名字
    • 位置参数变量是标准的数字
      • $0:程序名
      • $1:第一个参数
      • 29 9
$cat test
#!/bin/bash

total=$[ $1 * $2 ]
echo The total value is $total.
$./test 2 5
The total value is 10.
  • shell参数可以是字符串,每个参数都是用空格分隔的,所以shell会将空格当成分隔两个值的分隔符
    • 参数值中包含空格,必须要用引号(单引号双引号都可以)
#cat test
#!/bin/bash

echo Hello $1, glad to meet you
$./test 'Rich Blum'
Hello Rich Blum, glad to meet you.
$
  • 脚本需要多于9个命令行参数时,需要使用花括号(),比如,$10

13.1.2 读取程序名

  • $0:获取shell在命令行启动的程序的名字
    • 当传给$0变量的真实字符串是整个脚本的路径时,程序中就会使用整个路径,而不仅仅是程序名
    • basename命令会只返回程序名而不包括路径
$cat test
#!/bin/bash
name=`basename $0`
echo The command entered is : $0
echo The command entered is : name
$./test
The command entered is : ./test
The command entered is : test
$/home/rich/test
The command entered is : /home/rich/test
The command entered is : test
$

13.1.3 测试参数

  • 当脚本认为参数变量中有数据而实际上并没有时,会得到一个错误
    • 解决方法:在使用参数前检查参数[ -n “$1” ]

13.2 特殊参数变量

13.2.1 参数计数

  • $#特殊变量:含有脚本运行时就有的命令行参数的个数,可以在脚本中任何地方使用,跟普通变量一样
    • 最后一个参数的表示形式是!#而不是$#
    • 当命令行上没有任何参数时,#的值为0,在params变量中也为0,但!#变量返回命令行用到的脚本名
$cat test
#!/bin/bash

if [ $# -ne 2 ]
then 
    echo Usage: test a b
else
    total=$[ $1 + $2 ]
    echo The total is $total
fi
$./test 
Usage: test a b
$./test 10
Usage: test a b
$./test 10 15
The total is 25
$./test 10 15 20
Usage: test a b

13.2.2 抓取所有的数据

  • @变量提供了对所有参数的快速访问,这两个都能在单个变量中存储所有的命令行参数
    • $*:会将命令行上提供的所有参数当做单个单词保存,即当成一个参数
    • $@:将命令行上提供的所有参数当做同一字符串中的多个独立的单词,允许遍历所有的值,将提供的每个参数分隔开来,通常用for命令完成

13.3 移动变量

  • shift命令:根据它们的相对位置来移动命令行参数
    • 默认情况下会将每个参数变量减1,所以变量 3 2, 2 1,而变量$1的值会被删除
    • shift命令可以提供一个参数n来执行多位移动,如shift 2:连续移动2位
    • 当一个参数被移除后,它的值会被丢掉无法恢复
    • 变量$0的值,也就是程序名不会改变
    • 可以用shift命令遍历命令行参数,尤其在不知道到底有多少个参数的时候
$cat test
#!/bin/bash

count=1
while [ -n "$1" ]
do
    echo "Parameter #$count = $1"
    count=$[ $count + 1 ]
    shift
done
$
$./test rich barbara
parameter #1 = rich
Parameter #2 = barbara

13.4 处理选项

  • 选项:跟在单破折线后面的单个字母,能改变命令的行为

13.4.1 查找选项

  • 在命令行上,选项紧跟在脚本名之后,就跟命令行参数一样
  • 处理简单选项:在提取参数时,用case语句来判断参数是否被格式化成了选项
$cat test
#!/bin/bash

while [ -n "$1" ]
do
    case "$1" in
    -a) echo "Found the -a option";;
    -b) echo "Found the -b option";;
    *) echo "$1 is not an option";;
    esac
    shift
done
$
$./test -a -c
Found the -a option
-c is not an option
$
  • 分离参数和选项
    • shell会用双破折线来表明选项结束了,遇到双破折线之后,脚本会安全地将剩下的命令行参数当做参数来处理,而不是选项
$cat test
#!/bin/bash

while [ -n "$1" ]
do
    case "$1" in
    -a) echo "Found the -a option";;
    -b) echo "Found the -b option";;
    --) shift
            break;;
    *) echo "$1 is not an option";;
    esac
    shift
done

count=1
for param in $@
do
    echo "Parameter #$count: $param"
    count=$[ $count + 1 ]
done
$
$./test -a test1
Found the -a option
test1 is not an option
$./test -a -- test1
Found the -a option
Parameter #1: test1
  • 处理带值的选项
    • 当命令行选项要求额外的参数时,脚本必须能检测并能正确的处理
$cat test
#!/bin/bash

while [ -n "$1" ]
do
    case "$1" in
    -a) echo "Found the -a option";;
    -b) param="$2"
          echo "Found the -b option, with parameter value $param"
          shift 2;;
    -c) echo "Found the -c option";;
    --) shift
            break;;
    *) echo "$1 is not an option";;
    esac
    shift
done

count=1
for param in "$@"
do
    echo "Parameter #$count: $param"
    count=$[ $count + 1 ]
done
$
$./test -a -b test1 -d
Found the -a option
Found the -b option, with parameter value test1
-d is not an option
$
$./test -b test1 -a -d
Found the -b option, with parameter value test1
Found the -a option
-d is not an option
$
  • 如果将多个选项放进一个参数中时,它就不工作了
$ ./test -ac
-ac is not an option
$

13.4.2 使用getopt命令

  • getopt:识别命令行,从而在脚本中解析它们时更方便
    • 命令格式:getopt options optstring parameters
      • optstring:定义了命令行有效的选项字母,还定义了哪些选项字母需要参数值
        • 首先,在optstring中列出你要在脚本中用到的每个命令行选项字母
        • 然后,在每个需要参数值的选项字母后加一个冒号
    • getopt命令会给予你定义的optstring解析提供的参数
$getopt ab:cd -a -b test1 -cd test2 test3
-a -b test1 -c -d -- test2 test3
$
  • 如果指定了一个不在optstring中的选项,默认情况下,getopt命令会产生一条错误信息
    • 如果想忽略这条错误,在命令后加上-q选项
$getopt ab:cd -a -b test1 -cde test2 test3
getopt: invail option --e
-a -b test1 -c -d -- test2 test3
$
$getopt -q ab:cd -a -b test1 -cde test2 test3
-a -b test1 -c -d -- test2 test3
$
  • 在脚本中使用getopt
    • 可以在脚本中使用getopt来格式化输入给脚本的任何命令行选项
    • 方法:
      • 首先,用getopt命令生成的格式化后的版本来替换已有的命令行选项和参数,用set命令可以做到
        • set命令的选项之一是双破折号,它会将命令行参数替换成set命令的命令行的值
      • 然后,该方法将原始的脚本命令行参数传递给getopt命令
      • 之后再将getopt命令的输出传给set命令,用getopt格式化后的命令行参数来替换原始的命令行参数
    • 看起来如下:
set -- `getopts -q ab:cd "$@"`
$cat test
#!/bin/bash

set -- `getopt -q ab:c "$@"`
while [ -n "$1" ]
do
    case "$1" in
    -a) echo "Found the -a option";;
    -b) param="$2"
          echo "Found the -b option, with parameter value $param"
          shift ;;
    -c) echo "Found the -c option";;
    --) shift
            break;;
    *) echo "$1 is not an option";;
    esac
    shift
done

count=1
for param in "$@"
do
    echo "Parameter #$count: $param"
    count=$[ $count + 1 ]
done
$
$./test -a -b test1 -cd test2 test3
Found the -a option
Found the -b option, with parameter value 'test1'
Found the -c option
Parameter #1: 'test2'
Parameter #1: 'test3'

以上是关于:处理用户输入的主要内容,如果未能解决你的问题,请参考以下文章

在 ruby​​ 1.8.7 中处理不同类型的 utf 连字符

输入时格式化注册码

链式回调中的输入 - 破折号

如何将破折号输入保存为全局变量

将字符从输入转换为字符串

用空格替换文本区域中的逗号、破折号和输入键