shell变量赋值与环境

shell变量赋值与环境
今天,跟着架构师(凡爷)搞了一个代码量统计的脚本,我学到了以下几条:
1)命令在脚本里,只是字符串,较长的命令可以改为变量,引用去执行
2)潜意识我想到if,添加语句块,而凡爷却想到了grep -v,节省了一大段代码
3)echo '' > stat 他自然而然地加上这一行,让output按不同区块相隔开,结构清晰
4)他定位问题精准而快速,很是钦佩
 
今晚关于shell变量,我做了很多测试,将其整理如下,以便温故而知新。
 
#赋值#
# hours_per_day=24 seconds_per_hour=3600 days_per_week=7
# echo $hours_per_day $seconds_per_hour $days_per_week
24 3600 7
 readonly 使变量成为只读模式,而对它们赋值是被禁止的# readonly hours_per_day seconds_per_hour days_per_week # hours_per_day=2-bash: hours_per_day: readonly variable# echo $hours_per_day $seconds_per_hour $days_per_week24 3600 7

export [-fn] [name[=word]] ...  用于创建(空也可以)、修改、删除、打印“环境”(environment)
export -p
      The supplied names are marked for automatic export to the environment of subsequently executed commands.
  “提供的names被标记为自动导出给后续执行命令的环境。”
      If the -f option is given, the names refer to functions.
  如果指定-f,names引用函数。(即代表函数名称,函数名也是变量名)
  If no names are given, or if the -p option  is supplied,  a  list  of  all  names that are exported in this shell is printed. 
  如果没有names给出或者提供-p选项,所有names的一个list在这个shell里导出(列出)
  The -n option causes the export property to be removed from each name. 
  如果是-n选项,导致属性从每一个name中删除。(事实上并未删除,只是不会输出到后续指令的执行环境中,在当前shell中还是有)
  If a variable name is followed by =word, the value of the variable  is  set  to word.  export returns an exit status of 0 unless an invalid option is encountered,
      one of the names is not a valid shell variable name, or -f is supplied with a name that is not  a  func‐tion.
  如果变量name后跟=word,则变量的值设置为word,export返回状态码为0,除非遇到一个无效选项、其中一个name不是有效的shell变量名,或者-f后面提供不是name不是function。 
  export是将变量放进环境(environment)里,环境是一个name与word的简单列表,可供所有执行中的程序使用。
  新的进程会从其父进程集成环境,也可以在建立新的子进程之前修改它。
  如
  PATH=$PATH:/usr/local/bin  更新PATH  export PATH     导出它(发布它)  $export -p 显示当前环境
 
env [-i] [ var=value ...] [ command_name [ arguments... ] ]
 
 -i 忽略继承的环境,仅使用命令行上所给出的变量与值。
 未提供command_name时,显示环境中所有变量的name与value,否则使用命令行中提供的变量复制。
 
 如:
 $ cat test.sh
            echo $HOSTNAME
            env
   
    $$ env -i  HOSTNAME=30 PATH=/bin:/usr/bin/  name=abc sh test.sh (忽略继承环境,只使用命令行提供的变量与值)    30
    HOSTNAME=30
    name=abc
    PATH=/bin:/usr/bin/
    PWD=/home/zhangchunyang
    SHLVL=1
    _=/bin/env

   
    $ env HOSTNAME=30 PATH=/bin:/usr/bin/ sh test.sh | less (继承环境,同名的则覆盖)    30
    HOSTNAME=30
    PYENV_ROOT=/home/zhangchunyang/.pyenv
    SHELL=/bin/bash
    TERM=xterm
    HISTSIZE=5
    CATALINA_HOME=/usr/local/tomcat
    SSH_CLIENT=117.131.199.244 31299 22
    SYBASE_JRE7=/home/opt/sybase/shared/JRE-7_0_21_64BIT
    LUAJIT_LIB=/usr/local/lib
    PYTHON_EGG_CACHE=/tmp/.python-eggs
   
    注意:打印时,env不会正确地为环境变量值加上引号,以供重新输入到shell中,如果需要此功能,可使用export -p   
unset [-v] variable ... 
unset -f function ...
    从当前shell删除变量与函数  可以是任何变量(局部变量、环境变量)   
    -f: 解除(删除)指定的函数
    -v: 解除(删除)指定的变量。 没有任何选项时,这是默认行为模式
   
    如:        unset full_name            删除full_name        unset -v first middle last  删除这些变量     
        who_is_on () {
            who | awk '{print $1}' | sort -u
        }
        ...       
        unset -f who    删除函数
       
        $ cat test.sh
        a=1
        b=2
        c=3
        d=4
        e=5
        f=6
       
        who_is_on () {
            who | awk '{print $1}' | sort -u
        }       
        unset -v b       
        unset -f who_is_on       
        echo $a $b $c $d $e $f
        who_is_on
       
        $ sh test.sh
        1 3 4 5 6
        test.sh: line 14: who_is_on: command not found
       
        $ unset PATH
        $ ls
        -bash: ls: No such file or directory
        $ vim
        -bash: vim: No such file or directory
 
 declare [-aAfFgilrtux] [-p] [name[=value] ...]  #查看变量属性
 typeset [-aAfFgilrtux] [-p] [name[=value] ...] 声明/宣告 变量   
   
    -a    Each name is an indexed array variable (see Arrays above) 声明name是索引数组变量
    -A    Each name is an associative array variable (see Arrays above). 声明name是关联数组变量   
    -f    Use function names only. 使用函数名
    -i    The variable is treated as an integer; arithmetic evaluation (see ARITHMETIC EVALUATION above) is
              performed when the variable is assigned a value. 声明整型变量。当变量分配值时被表现为算术求值。   
    -l    When the variable is assigned a value, all upper-case characters  are  converted  to  lower-case.
              The upper-case attribute is disabled. 禁用大写,分配的大写自动转为小写
    -r    Make  names readonly.  These names cannot then be assigned values by subsequent assignment state‐
              ments or unset. 设置为只读变量。这些names不能被后续再分配状态(修改)和删除(unset)
    -t    Give each name the trace attribute.  Traced functions inherit the DEBUG and RETURN traps from the
              calling shell.  The trace attribute has no special meaning for variables.  给name trace属性
    -u    When  the  variable  is  assigned a value, all lower-case characters are converted to upper-case.
              The lower-case attribute is disabled. 禁用小写,分配的值为小写自动转为大写
    -x    Mark names for export to subsequent commands via the environment. 标记names为export给后续命令的环境
            -x 即export 发布的变量           
           
            # export -p | less
            declare -x HISTCONTROL="ignoredups"
            declare -x HISTSIZE="1000"
            declare -x HISTTIMEFORMAT="%F %T:%t"
            declare -x HOME="/root"
            declare -x HOSTname="Webserver"
            ...
           
            $ declare -u f
            $ f='sdafsadf'
            $ declare -u           
            declare -u f="SDAFSADF"
       
        上面这些属性是可以叠加的
            如declare -irx Uid="0"参数展开(parameter expansion)是shell提供变量值在程序中使用的过程。

1) #引用变量值#    # reminder='Time to go to the dentist!'
    # echo $reminder
    Time to go to the dentist!   
    # echo _${reminder}_  #使用{}可以保护变量,避免额外的字符影响变量。
    _Time to go to the dentist!_
   
    警告: 默认,未定义的变量展开为null(空字符串)。程序随便乱写,就可能会导致灾难发生:           
                # rm -fr /$MYPROGRAM        如未设置MYPROGRAM,就会有大灾难发生了!
        所以,写程序要非常小心。2) #替换运算符# 测试字符串变量存在状态,且某种情况下允许默认值替换。

运算符替换例子用途
${varname:-word}如果varname存在且非null,则返回其值;否则,返回word# echo ${var:-hello,world} var为不为空,返回其值;否则,返回hello,world如果变量未定义,则返回默认值
${varname:=word}如果varname存在且非null,则返回它的值;否则,设置它为word,并返回其值# echo ${var:-hello,world} ;如果var有值,返回其值;否则,返回hello,world,并赋值给var如果变量未定义,则设置变量为默认值
${varname:?message}如果varname存在且非null,则返回它的值;否则,显示varname:message,并退出当前的命令或脚本。省略message,会出现默认信息parameter null or not set。${count:?"undefined/1"}将显示 count:undefined!,如果count未定义,则退出为了捕捉由于变量未定义所导致的错误。
${varname:+word}如果varname存在且非null,则返回word;否则,返回null。如果count已定义,则${count:+1}返回1(也就是“真”)为测试变量存在

3) #模式匹配运算符#
    模式为`通配符(wildcard)`
    基本通配符:
        ?      任何单一字符
        *      任何的字符字符串
        [set]  任何在set里的字符
        [!set]  任何不在set里的字符
    如:
        [abc]
        [.,;] 句点,逗点或分号(awk 多字符分割的写法)
        [-_]  破折号或下划线
        [a-c]
        [a-z]
        [!0-9] 任何一个非数字字符
        [0-9!] 任何一个数字或惊叹号
        [a-zA-Z0-9_-]
${variable#pattern} 删除variable从左到右最短匹配
${variable##pattern} 删除variable从左到右最长匹配
${variable%pattern} 删除variable从右到左最短匹配
${variable%%pattern}删除variable从右到左最长匹配
附加:
    git不熟,我只能帮凡爷最后一部分,他那部分未经他允许,还是不放了。

for name in `ls $respath |grep -v gitresult`
do
    sort -u $respath/$name | while read commit ; do git show $commit --pretty=tformat: --numstat;done  | \
    awk -v user=$name '{ add += $1; subs += $2; loc += $1 - $2 } END { printf "%s added lines: %s, removed lines: %s, total lines: %s\n", user ,  add, subs, loc }' >> $respath/gitresult.$since.$before
done
 
-------小技巧------
path=$1
respath=/home/zcy
respath="$respath/${path:-result}"
 
如果$1为空,则respath=/home/zcy/result;否则,respath=/home/zcy/$path
 
或者:
path=$1
path=${path:=result}
respath=/home/zcy/$path

相关推荐