1.如何编写简单的shell程序
什么是shell 脚本 最简单的解释,一个shell 脚本就是一个包含一系列命令的文件。
shell 读取这个文件,然后执行文件中的所有命令,就好像这些命令已经直接被输入到了命令行中一样。Shell 有些独特,因为它不仅是一个功能强大的命令行接口, 也是一个脚本语言解释器。
我们将会看到,大多数能够在命令行中完成的任务也能够用脚本来实现,同样地,大多数能用脚本实现的操作也能够在命令行中完成。虽然我们已经介绍了许多shell 功能,但只是集中于那些经常直接在命令行中使用的功能。
Shell 也提供了一些通常(但不总是)在编写程序时才使用的功能。怎样编写一个shell 脚本为了成功地创建和运行一个shell 脚本,我们需要做三件事情:1. 编写一个脚本。
Shell 脚本就是普通的文本文件。所以我们需要一个文本编辑器来书写它们。
最好的文本编辑器都会支持语法高亮,这样我们就能够看到一个脚本关键字的彩色编码视图。语法高亮会帮助我们查看某种常见错误。
为了编写脚本文件,vim,gedit,kate,和许多其它编辑器都是不错的候选者。2. 使脚本文件可执行。
系统会相当挑剔不允许任何旧的文本文件被看作是一个程序,并且有充分的理由! 所以我们需要设置脚本文件的权限来允许其可执行。3. 把脚本放置到shell 能够找到的地方当没有指定可执行文件明确的路径名时,shell 会自动地搜索某些目录,来查找此可执行文件。
为了最大程度的方便,我们会把脚本放到这些目录当中。脚本文件格式 为了保持编程传统,我们将创建一个“hello world”程序来说明一个极端简单的脚本。
所以让我们启动我们的文本编辑器,然后输入以下脚本:#!/bin/bash# This is our first script.echo 'Hello World!' 对于脚本中的最后一行,我们应该是相当的熟悉,仅仅是一个带有一个字符串参数的echo 命令。对于第二行也很熟悉。
它看起来像一个注释,我们已经在许多我们检查和编辑过的配置文件中看到过。关于shell 脚本中的注释,它们也可以出现在文本行的末尾,像这样:echo 'Hello World!' # This is a comment too文本行中,# 符号之后的所有字符都会被忽略。
类似于许多命令,这也在命令行中起作用:[me@linuxbox ~]$ echo 'Hello World!' # This is a comment tooHello World! 虽然很少在命令行中使用注释,但它们也能起作用。 我们脚本中的第一行文本有点儿神秘。
它看起来它应该是一条注释,因为它起始于一个# 符号,但是它看起来太有意义,以至于不仅仅是注释。事实上,这个#! 字符序列是一种特殊的结构叫做shebang。
这个shebang 被用来告诉操作系统将执行此脚本所用的解释器的名字。每个shell 脚本都应该把这一文本行作为它的第一行。
让我们把此脚本文件保存为hello_world。可执行权限下一步我们要做的事情是让我们的脚本可执行。
使用chmod 命令,这很容易做到:[me@linuxbox ~]$ ls -l hello_world-rw-r--r-- 1 me me 63 2009-03-07 10:10 hello_world[me@linuxbox ~]$ chmod 755 hello_world[me@linuxbox ~]$ ls -l hello_world-rwxr-xr-x 1 me me 63 2009-03-07 10:10 hello_world对于脚本文件,有两个常见的权限设置;权限为755 的脚本,则每个人都能执行,和权限为700 的脚本,只有文件所有者能够执行。注意为了能够执行脚本,脚本必须是可读的。
脚本文件位置当设置了脚本权限之后,我们就能执行我们的脚本了:[me@linuxbox ~]$ ./hello_worldHello World!为了能够运行此脚本,我们必须指定脚本文件明确的路径。如果我们没有那样做,我们会得到这样的提示:[me@linuxbox ~]$ hello_worldbash: hello_world: command not found为什么会这样呢?什么使我们的脚本不同于其它的程序?结果证明,什么也没有。
我们的脚本没有问题。是脚本存储位置的问题。
我们了解到PATH 环境变量及其它在系统查找可执行程序方面的作用。回顾一下,如果没有给出可执行程序的明确路径名,那么系统每次都会搜索一系列的目录,来查找此可执行程序。
这个/bin 目录就是其中一个系统会自动搜索的目录。这个目录列表被存储在一个名为PATH 的环境变量中。
这个PATH 变量包含一个由冒号分隔开的目录列表。我们可以查看PATH 的内容:[me@linuxbox ~]$ echo $PATH/home/me/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games这里我们看到了我们的目录列表。
如果我们的脚本驻扎在此列表中任意目录下,那么我们的问题将会被解决。注意列表中的第一个目录,/home/me/bin。
大多数的Linux 发行版会配置PATH 变量,让其包含一个位于用户家目录下的bin 目录,从而允许用户能够执行他们自己的程序。所以如果我们创建了一个bin 目录,并把我们的脚本放在这个目录下,那么这个脚本就应该像其它程序一样开始工作了:[me@linuxbox ~]$ mkdir bin[me@linuxbox ~]$ mv hello_world bin[me@linuxbox ~]$ hello_worldHello World!它的确工作了。
如果这个PATH 变量不包含这个目录,我们能够轻松地添加它,通过在我们的.bashrc 文件中包含下面这一行文本:export PATH=~/bin:"$PATH"当做了这个修改之后,它会在每个新的终端会话中生效。为了把这个修改应用到当前的终端会话中。
2.怎么写LINUX的SHELL
在进行linux测试时编写脚本是必不可少的,Shell脚本的名称可以随便定义,也不要什么后缀名,例如可以写abc,smartzip这类名称,运行时只要键入 ./smartzip就能运行脚本了。
每行命令开头处不用就空格也行。
第1部分. Linux 脚本编写基础 1.1 语法基本介绍 1.1.1 开头 程序必须以下面的行开始(必须方在文件的第一行): #!/bin/sh 符号#!用来告诉系统它后面的参数是用来执行该文件的程序。
在这个例子中我们使用/bin/sh来执行程序。 当编辑好脚本时,如果要执行该脚本,还必须使其可执行。
要使脚本可执行: 编译 chmod +x filename 这样才能用./filename 来运行 1.1.2 注释 在进行shell编程时,以#开头的句子表示注释,直到这一行的结束。我们真诚地建议您在程序中使用注释。
如果您使用了注释,那么即使相当长的时间内没有使用该脚本,您也能在很短的时间内明白该脚本的作用及工作原理。 1.1.3 变量 在其他编程语言中您必须使用变量。
在shell编程中,所有的变量都由字符串组成,并且您不需要对变量进行声明。要赋值给一个变量,您可以这样写: #!/bin/sh #对变量赋值: a="hello world" # 现在打印变量a的内容: echo "A is:" echo $a 有时候变量名很容易与其他文字混淆,比如: num=2 echo "this is the $numnd" 这并不会打印出"this is the 2nd",而仅仅打印"this is the ",因为shell会去搜索变量numnd的值,但是这个变量时没有值的。
可以使用花括号来告诉shell我们要打印的是num变量: num=2 echo "this is the ${num}nd" 这将打印: this is the 2nd 1.1.4 环境变量 由export关键字处理过的变量叫做环境变量。我们不对环境变量进行讨论,因为通常情况下仅仅在登录脚本中使用环境变量。
1.1.5 Shell命令和流程控制 在shell脚本中可以使用三类命令: 1)Unix 命令: 虽然在shell脚本中可以使用任意的unix命令,但是还是由一些相对更常用的命令。这些命令通常是用来进行文件和文字操作的。
常用命令语法及功能 echo "some text": 将文字内容打印在屏幕上 ls: 文件列表 wc –l file :计算文件行数 wc -w file:计算文件中的单词数 wc -c file:计算文件中的字符数 cp sourcefile destfile: 文件拷贝 mv oldname newname : 重命名文件或移动文件 rm file: 删除文件 grep 'pattern' file: 在文件内搜索字符串比如:grep 'searchstring' file.txt cut -b colnum file: 指定欲显示的文件内容范围,并将它们输出到标准输出设备比如:输出每行第5个到第9个字符cut -b5-9 file.txt千万不要和cat命令混淆, 这是两个完全不同的命令 cat file.txt: 输出文件内容到标准输出设备(屏幕)上 file somefile: 得到文件类型 read var: 提示用户输入,并将输入赋值给变量 sort file.txt: 对file.txt文件中的行进行排序 uniq: 删除文本文件中出现的行列比如: sort file.txt | uniq expr: 进行数学运算Example: add 2 and 3expr 2 "+" 3 find: 搜索文件比如:根据文件名搜索find . -name filename -print tee: 将数据输出到标准输出设备(屏幕) 和文件比如:somecommand | tee outfile basename file: 返回不包含路径的文件名比如: basename /bin/tux将返回 tux dirname file: 返回文件所在路径比如:dirname /bin/tux将返回 /bin head file: 打印文本文件开头几行 tail file : 打印文本文件末尾几行 sed: Sed是一个基本的查找替换程序。可以从标准输入(比如命令管道)读入文本,并将 结果输出到标准输出(屏幕)。
该命令采用正则表达式(见参考)进行搜索。不要和shell中的通配符相混淆。
比如:将linuxfocus 替换为LinuxFocus :cat text.file | sed 's/linuxfocus/LinuxFocus/' > newtext.file awk: awk 用来从文本文件中提取字段。缺省地,字段分割符是空格,可以使用-F指定其他分割符。
cat file.txt | awk -F, '{print $1 "," $3 }'这里我们使用,作为字段分割符,同时打印第一个和第三个字段。如果该文件内容如下: Adam Bor, 34, IndiaKerry Miller, 22, USA命令输出结果为:Adam Bor, IndiaKerry Miller, USA 2) 概念: 管道, 重定向和backtick 这些不是系统命令,但是他们真的很重要。
管道 (|) 将一个命令的输出作为另外一个命令的输入。 grep "hello" file.txt | wc -l 在file.txt中搜索包含有”hello”的行并计算其行数。
在这里grep命令的输出作为wc命令的输入。当然您可以使用多个命令。
重定向:将命令的结果输出到文件,而不是标准输出(屏幕)。 > 写入文件并覆盖旧文件 >> 加到文件的尾部,保留旧文件内容。
反短斜线 使用反短斜线可以将一个命令的输出作为另外一个命令的一个命令行参数。 命令: find . -mtime -1 -type f -print 用来查找过去24小时(-mtime –2则表示过去48小时)内修改过的文件。
如果您想将所有查找到的文件打一个包,则可以使用以下脚本: #!/bin/sh # The ticks are backticks (`) not normal quotes ('): tar -zcvf lastmod.tar.gz `find . -mtime -1 -type f -print` 3)流程控制 1.if "if" 表达式 如果条件为真则执行then后面的部分: if 。.; then 。
. elif 。.; then 。
. else 。. fi 大多数情况下,可以使用测试命令来对条件进行测试。
比如可以比较字符串、判断文件。
3.编写shell程序
1编写一个shell程序,该程序可以在某一目录下查找一指定文件,如果文件存在,则把该文件复制到用户自己创建的子目录中。
#!/bin/shif [ -z "$1" ] || [ ! -f $1 ];then echo -e "文件不存在!"else if [ -z "$2" ];then echo -e "请输入要创建的文件夹名称" else if [ -d "$2" ];then cp ./$1 ./$2/ echo -e "拷贝成功!" else mkdir $2 cp ./$1 ./$2/ echo -e "拷贝成功!" fi fifi2 编写一个shell程序,该程序可以查找所有的登陆用户,并修改指定用户的权限。这个需要root权限的用户才能操作修改权限哦,查找所有登录用户命令是 who -u修改权限命令是chown。
转载请注明出处育才学习网 » 怎么写shell程序