zoukankan      html  css  js  c++  java
  • 《Shell脚本学习指南》第二章学习笔记


    2.1 脚本编程语言与编译型语言的差异

    编译型语言从源代码转换成目标代码,便能直接通过计算机执行。好处是高效,但运作于底层。
    例如,在C++里,很难进行“将一个目录里所有的文件复制到另一个目录中”之类的简单操作。

    脚本编程语言通常是解释型(Interpreted)的。由解释器读入程序代码,并将其转换成内部的形式,再执行。
    解释器本身是一般的编译型程序。


    2.2 为什么要使用Shell脚本

    脚本执行效率通常不如编译型语言,但它已经足够快了,足以忽略它性能上的问题。
    花一个小时写成的简单脚本,同样的功能用C或C++来编写实现,可能需要两天。
    脚本编程语言的例子有awk、Perl、Python、Ruby与Shell。

    因为Shell是各Unix系统之间通用的功能,并且经过了POSIX的标准化。只要用心写一次,
    即可应用到很多系统中。


    2.3 一个简单的脚本

    自动化统计当前有多少用户登录系统。其中用到了who命令,wc -l统计行数以及管道。

    $ cat > nusers          建立文件,使用cat复制终端的输入
    who | wc -l
    ^D                         Ctrl-D表示end-of-file
    $ chmod +x nusers    让文件拥有执行的权限
    $ ./nusers                执行测试
         6

    这展现了小型Shell脚本的典型开发周期:
    1. 直接在命令行上测试,找到能够完成工作的适当语法。
    2. 将它们放进一个独立的脚本里,并为该脚本设置执行的权限。
    3. 之后就能直接使用该脚本了。


    2.4 自给自足的脚本:位于第一行的#!

    当一个文件中开头的两个字符是#!时,内核会扫描该行其余的部分,看是否存在
    可用来执行程序的解释器的完整路径。此外还会扫描是否有一个选项要传给解释器。

    #! /bin/csh -f

    下面是几个初级的陷阱:
    1. 各系统对#!这一行的长度限制从64到1024字符都有,所以尽量不要超过64个字符。
    2. 别在选项之后放置任何空白,因为空白也会跟着选项一起传递给被引用的程序。


    2.5 Shell的基本元素

    2.5.1 命令与参数

    命令名称是命令行的第一个项目,后面跟着选项。选项可有可无,还可以合并,如ls -lt比ls -l -t更方便。
    长选项在标准工具GNU版本,以及X Window System中使用的越来越普遍,如patch --verbose --backup。
    分号;可用来分隔同一行里的多条命令,Shell会依次执行这些命令。
    用&符号,则Shell将在后台执行其前面的命令,Shell不用等待该命令完成就可以继续执行下一个命令。

    Shell识别三种基本命令:内建命令、Shell函数以及外部命令

    1.内建命令就是由Shell本身所执行的命令。有些命令是由于必要性(需要改变父Shell进程的属性)才内建的。
    如cd用来改变目录,read将用户输入传给Shell变量。另外一些是为了效率,如test命令,还有I/O命令echo与printf。

    2.Shell函数用Shell语言写成的,可以像命令那样引用的代码。

    3.外部命令就是Shell子进程所执行的命令,基本过程如下:
    a.新建一个父Shell进程的子进程。(fork)
    b.在新进程里搜索PATH变量列出的目录,找到特定命令。当命令名称含有斜杠/时,略过路径查找步骤。
    c.在新进程里,用找到的程序取代执行中的Shell程序并执行。(exec)
    d.程序完成后,父Shell进行继续从终端读取下一条命令,或执行脚本里的下一条命令。

    2.5.2 变量

    Shell变量名称的开头是一个字符或下划线,后面可以接着任意长度的字母、数字或下划线。
    变量名称和变量保存的字符串的长度都没有限制。在变量名称前加$字符取出Shell变量的值。
    当给变量赋予的值包含空格时,需要加上引号。

    $ myvar=this_is_a_long_string
    $ echo $myvar

    $ first=issac middle=bashevis last=singer     单行可进行多次赋值
    $ fullname="issac bashevis singer"
    $ fullname="$first $middle $last"
    $ oldname=$fullname

    2.5.3 简单的echo输出

    $ echo Now is the time for all good men
    Now is the time for all good men

    echo有版本上的差异,但只要是使用最简单的形式,其echo的可移植性不会有问题。

    BSD版本的echo,-n参数会省略结尾的换行符号。
    $ echo -n "Enter your name: "
    Enter your name: _

    System V的echo会处理转义字符,\c表示忽略换行符号。
    $ echo "Enter your name: \c"

    2.5.4 华丽的printf输出

    printf命令模仿C程序库里的printf()程序,几乎复制了该函数的所有功能。
    $ printf "Hello, world\n"
    $ printf "The first program always prints '%s, %s'!\n" Hello world
    最大不同在于:printf不像echo那样自动提供一个换行符,必须显式指定\n。

    2.5.5 基本的I/O重定向

    标准输入/输出可能是软件设计原则里最重要的概念了。程序不必知道也不用关心
    它的输入与输出背后是什么设备:磁盘上的文件、终端、磁带机、网络连接或是
    另一个执行中的程序!当程序启动时,可以预期的是,标准输入输出都已打开,
    且已准备好供其使用。

    许多Unix程序都遵循这一设计原则。默认情况下,它们会读取标准输入、写入标准
    输出,并将错误信息传递到标准错误输出。这类程序常叫做过滤器(filter)
    默认的标准输入、标准输出以及标准错误输出都是终端。

    $ cat                  未指定任何参数,读取标准输入,写入标准输出
    now is the time     由用户输入     
    now is the time     由cat返回
    for all good men
    for all good men
    ^D

    在你登录时,Unix便将默认的标准输入、输出及错误输出安排成你的终端。

    重定向与管道

    以<改变标准输入:tr -d '\r' < dos-file.txt (命令tr -d '\r'将删除回车)
    以>改变标准输出,覆盖目的文件:tr -d '\r' < dos-file.txt > UNIX-file.txt
    以>>附加到文件,附加到文件结尾:
    for f in dos-file*.txt
    do
         tr -d '\r' >> big-UNIX-file.txt
    done

    以|建立管道:tr -d '\r' < dos-file.txt | sort > UNIX-file.txt
    将program1的标准输出修改为program2的标准输入。虽然<与>可将输入与输出连接到文件,
    不过管道的执行速度比使用临时文件的程序快上十倍。

    构造管道时,应该试着让每个阶段的数据量变少。换句话说,如果你有两个要完成的步骤与
    先后次序无关,你可以把让数据量变少的那一步放在管道的前端,这样可以提升脚本的性能。
    例如,使用sort排序之前,先以grep找出相关的行,这样可以让sort少做些事。

    特殊文件:/dev/null与/dev/tty

    /dev/null就是大家所熟知的位桶(bit bucket),写到此文件的数据都会被系统丢掉。读取它
    则会立即返回文件结束符号(end-of-file)。

    当程序打开/dev/tty时,Unix会自动将它重定向到一个终端,这在程序必须读取人工输入时特别有用。
    以下stty命令用来控制终端的各种设置。
    $ printf "Enter new password: "     提示输入
    $ stty -echo                              关闭自动打印输入字符的功能
    $ read pass < /dev/tty               读取密码
    $ printf "\nEnter again: "              提示再输入一次
    $ read pass2 < /dev/tty              再读取一次以确认
    $ stty echo                               恢复自动打印输入字符的功能
    $ echo
    $ echo pass is $pass, pass2 is $pass2

    2.5.6 基本命令查找

    如果要编写自己的脚本,最好准备自己的bin目录来存放它们,并且让Shell能自动找到它们。
    只要建立自己的bin目录,并将它加入$PATH中的列表即可。
    $ PATH=$PATH:$HOME/bin

    要让修改永久生效,在.profile文件中把你的bin目录加入$PATH,每次登录时Shell都会读取
    .profile文件。

    空项目或点号表示当前目录,如PATH=:/bin:/user/bin或PATH=/bin::/user/bin


    2.6 访问Shell脚本的参数

    Shell脚本的命令行参数,同时也是函数的参数。各参数都由整数来命名,超过9就应该用大括号。
    echo first arg is $1
    echo tenth arg is {$10}

    查找某用户现在是否登录。
    $ cat > finduser
    #! /bin/sh
    # finduser -- find if user in argument one is login now
    who | grep $1
    ^D
    $ chmod +x finduser
    $ ./finduser betsy


    2.7 简单的执行跟踪

    打开执行跟踪功能会使得Shell显示每个被执行到的命令,并在前面加上“+”和一个空格。
    $ sh -x nusers
    + who
    + wc -l

    也可以在脚本里,用set -x命令打开执行跟踪,然后再用set +x命令关闭。
    $ cat > trace1.sh
    #! /bin/sh
    set -x
    echo 1st echo
    set +x
    echo 2nd echo
    ^D
    $ chmod +x trace1.sh
    $ ./trace1.sh
    + echo 1st echo
    1st echo
    + set +x
    2nd echo


  • 相关阅读:
    树莓派关闭swap文件[浙大嵌入式系统]
    树莓派之objdump [浙大嵌入式系统]
    树莓派RAM[浙大嵌入式系统]
    温湿度传感器dht11的lcd显示[嵌入式系统]
    ucos II[嵌入式系统]
    MooTools 1.4 源码分析 Fx.CSS
    MooTools 1.4 源码分析 (关于Core、Type等模块分析)
    MooTools 1.4 源码分析 Fx.Tween
    MooTools Class.Mutators 如何建立一个我们自己的Mutator
    Javascript各种循环测试,达夫设备可用性分析
  • 原文地址:https://www.cnblogs.com/xiaomaohai/p/6157876.html
Copyright © 2011-2022 走看看