关于shell命令的相关事例操作

Shell是系统的用户界面,提供了用户与内核进行交互操作的一种接口。它接收用户输入的命令并把它送入内核去执行。

Linux中的shell有多种类型,其中最常用的几种是Bourne shell(sh)、C shell(csh)和Korn shell(ksh)
三种shell各有优缺点。Bourne shell是UNIX最初使用的shell,并且在每种UNIX上都可以使用。Bourne shell在shell编程方面相当优秀,但在处理与用户的交互方面做得不如其他几种shell。Linux操作系统缺省的shell是Bourne Again shell,它是Bourne shell的扩展,简称Bash,与Bourne shell完全向后兼容,并且在Bourne shell的基础上增加、增强了很多特性。Bash放在/bin/bash中,它有许多特色,可以提供如命令补全、命令编辑和命令历史表等功能,它还包含了很多C shell和Korn shell中的优点,有灵活和强大的编程接口,同时又有很友好的用户界面。

总的来说相当于windows下的bat执行语句。

shell语句的特点
1.文件必须以sh结尾;
2.文件默认产生出来的没有可执行权限,可通过chmod来修改权限;

3.vi分为指令模式和输入模式

a从光标所在位置之后面开始心中内容
A从光标在列的第一个非空字节签名开始插入内容
在指令模式下::q没有修改退出 q!放弃保存退出 wq、x保存退出
/关键字 查找

例子1判断分数

流程

1.在当前文件夹下新建一个shell文件vi score.sh,然后进入vi编辑模式

输入代码

echo "iput s:"
read s
echo "iput is $s"
if [ $s -ge 80 ] ; then
echo good
elif [ $s -ge 60 ] ; then
echo pass
else
echo fail
fi

按ESC,进入命令编辑模式,再按一下shift,再按一下shift+z就快速保存退出

2.在默认条件下,执行shell文件会出现permission denied报错,一般是没有可执行权限。用chmod修改权限

chomd 777 score.sh //把所有权限给aa文件 777代表所有权限

3.接着使用 ./score.sh就可以执行shell了

案例2,:一个小型超市查询系统

实现功能:1.使用账户名adimin 密码123 登陆系统 2.选择要使用的服务,包括查询所以产品、查询单个产品信息、退出。

echo "welcome to supermarket system , please choose: ?"
select var in "Login" "Exit" ; do
break
done
flag=0
while [ $flag -eq 0 ]
do
if [ $var = "Login" ]; then
echo "Please input your name:"
read name
echo "please input your password"
read pass
if [ $name = "admin" -a $pass = "123" ];then
flag=1
echo "Please choose your option:"
select var in "view all products" "view information about a product" "Exit"; do
break
done
if [ $var = "view information about a product" ];then
echo "Please input the product name:"
read fname
if [ $fname = "fish" ];then
echo "fish price 10.0 amount 100kg"
elif [ $fname = "pork" ];then
echo "pork price 8.0 amount 120kg"
elif [ $fname = "tomato" ];then
echo "tomato price 3.0 amount 100kg"
else
echo "no such things"
fi
fi
fi
fi
done


在 Linux 系统中,当你输入一个命令,再按两次TAB 键,就会列出所有以你输入字符开头的可用命令。这并不新鲜,可能你已经知道了。这个功能被称作命令行补全bash completion。默认情况下,bash 命令行可以自动补全文件或目录名称。不过,我们可以增强 bash 命令补全功能,通过 complete 命令让它达到新的高度。
这个教程说明了我们是怎样使用可编程的命令行补全功能(programmable completion)把自动补全功能应用于选项或者命令行参数。
例如:在输入 write 命令之后,如果你按两次 TAB 按键,自动补全功能会提供可供执行 write 操作的用户列表。
$ write [TAB][TAB]
bala raj
jason randy
john ritu
mayla thomas
nisha www-data

在下面的例子中,可以为 telnet 命令显示可用的主机名:
$ telnet [TAB][TAB]
localhost dev-db fileserver

要让可编程命令补全功能在你的终端起作用 ,你只需要如下执行/etc/bash_completion即可:
# . /etc/bash_completion
你也可以取消/etc/bash.bashrc(来自 Ubuntu Linux 13.04 系统)中如下的注释,这样,你就可以不需要执行上面的命令了:
### enable bash completion in interactive shells
if ! shopt -oq posix; then
if [ -f /usr/share/bash-completion/bash_completion ]; then
. /usr/share/bash-completion/bash_completion
elif [ -f /etc/bash_completion ]; then
. /etc/bash_completion
fi
fi

如果你没有发现这些代码,也没有找到/etc/bash_completion文件,那么你只需要通过使用apt-get命令来安装bash_completion 包即可。

1. 查看已有的命令行补全
在启用可编程的命令行补全功能后,就已经有了一些定义好的命令补全功能。complete 命令用于定义命令行补全。
要查看已有的命令行补全,如下使用 complete 命令:
complete -p | less
上面例子中的 -p 选项是可选的。
2. 列出 bash 中标准补全功能
默认情况下,Bash 为 Linux 用户提供了下列标准补全功能。
1.变量补全
2.用户名补全
3.主机名补全
4.路径补全
5.文件名补全
我们在之前的 bash 标准补全中讨论过这些。
3. 定义一个命令名补全
通过 -c 选项可以将所有的可用命令作为一个命令的补全参数。在下面的例子里面,为 which 命令定义了一个补全(LCTT译注:在按两下 TAB 时,可以列出所有命令名作为可补全的参数)。

$ complete -c which

$ which [TAB][TAB]
Display all 2116 possibilities? (y or n)

如上,如果按下 ‘y',就会列出所有的命令名。
4. 定义一个目录补全
通过选项 -d,可以定义一个仅包含目录名的补全参数。在下面的例子中,为 ls 命令定义了补全。

$ ls
countfiles.sh dir1/ dir2/ dir3/

$ complete -d ls

$ ls [TAB][TAB]
dir1/ dir2/ dir3/

如上,连按下 TAB 仅会显示目录名。
5. 定义一个后台任务名补全
补全功能也能够以任务名作为补全参数。选项 -j 可以定义任务名作为传递给命令的参数,如下:
$ jobs
[1]- Stopped cat
[2]+ Stopped sed 'p'

$ complete -j ./list_job_attrib.sh

$ ./list_job_attrib.sh [TAB][TAB]
cat sed

关于后台任务,你可以参考 Linux 后台任务中的例子了解如何管理后台任务。
6. 带有前缀和后缀的补全
补全功能可以为实际的补全内容定义前缀和后缀。在下面的例子中,为 list_job_attrib.sh 定义了补全内容的前缀和后缀。
$ jobs
[1]+ Stopped cat
$ complete -P '">' -S '<"' ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]
$ ./list_job_attrib.sh ">cat<"

7. 带有排除的文件名和目录名补全
假如脚本运行完成后,输出目录如下:
$ cd output/

$ ls
all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt
parser_mod.tmp extract.o
如上,如果你想要 ls 命令的补全忽略 .tmp 和 .o 文件:

$ export FIGNORE='.tmp:.o'
$ complete -f -d ls
$ cd output
$ ls [TAB][TAB]
all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt

FIGNORE 是一个环境变量,它包含了自动补全所需要排除的文件名后缀。
8. 通过 IFS 变量分割字符串得到补全值
可以通过 -W 选项定义补全值列表,然后通过 IFS 环境变量进行切分。切分结果会展开变量并作为补全显示。
$ export IFS=" "
$ complete -W "bubble quick" ./sort_numbers.sh
$ ./sort_numbers.sh [TAB][TAB]
bubble quick

如上所述,字符串通过 IFS 分隔符进行切分后,内嵌的变量会被展开为变量值,所以可以如下使用变量:
$ echo $SORT_TYPE1
bubble

$ echo $SORT_TYPE2
quick

$ complete -W "$SORT_TYPE1 $SORT_TYPE2" ./sort_numbers.sh
$ ./sort_numbers.sh [TAB][TAB]
bubble quick

9. 写个函数来生成补全
你可以引入一个函数来定义补全。使用 -F 选项将函数名传给 complete 命令,执行函数生成补全内容。例如,函数如下:
_parser_options()
{
local curr_arg;
curr_arg=${COMP_WORDS[COMP_CWORD]}
COMPREPLY=( $(compgen -W '-i --incoming -o --outgoing -m --missed' -- $curr_arg ) );
}
在上述函数中:
1.COMPREPLY : 该数组控制连按下 TAB 后显示的结果
2.COMP_WORDS : 该数组包含命令行输入的单词
3.COMP_CWORD : COMP_WORDS 数组的索引,使用它来区分命令行可以访问的单词位置
4.compgen : -W 基于 $current_arg 提供可能的补全及其参数
该函数放在 parser_option 文件中,并通过 source 命令引入:
$ source parser_option
将该函数和你的 parser.pl 脚本关联起来:
$ complete -F _parser_options ./parser.pl
$ ./parser.pl [TAB][TAB]
-i --incoming -o --outgoing -m --missed
如上,parser.pl 的选项是由函数 _parser_options() 生成的。
提示: 查看/etc/bash_completion 来了解更多的可编程补全函数。
10. 当第一个规则没有生成结果时,就使用第二个
如果定义的补全规则没有生成匹配时,可以使用 -o 选项生成补全。
$ complete -F _count_files -o dirnames ./countfiles.sh
如上,为 ./countfiles.sh 定义了 _count_files 补全函数。 如果 the _count_files() 函数没有生成任何匹配的话,就会触发目录补全。
$ ls
countfiles.sh dir1/ dir2/ dir3/

$./countfiles.sh [TAB][TAB]
dir1 dir2 dir3

 

posted @ 2017-10-08 18:43  Lexone  阅读(360)  评论(0)    收藏  举报