美文网首页Linux 相关文章程序员
Zsh 开发指南(第十一篇 变量的进阶内容)

Zsh 开发指南(第十一篇 变量的进阶内容)

作者: 陌辞寒 | 来源:发表于2017-08-30 17:02 被阅读134次

    导读

    之前我们已经依次讲过 zsh 下的五种变量(字符串、数组、哈希表、整数、浮点数)的基本用法。但变量的使用方面,还有一些比较进阶的内容,这对一些比较特别的场景很有帮助。

    typeset 命令

    typeset 命令用于对变量进行详细的设置。我们之前在哈希表那篇见过它。typeset -A 可以用来定义哈希表。

    % typeset -A hashmap=(aa bb cc dd)
    

    但我们后续都使用 local,因为 local 的功能和 hashmap 是一样的(除了不能用 -f 和 -g,这两个选项不常用),并且更短更容易输入。这里提到 typeset 命令,因为这个名称很好地反映了它的功能。但知道了这个后,我们可以继续使用 local 命令,毕竟它们是一样的。

    typeset 命令有很多选项,可以作用在变量上,起到各种各样的效果。

    强制字符串内容为小写或者大写

    # 强制字符串内容为小写
    % local -l str=abcABC && echo $str
    abcabc
    
    # 强制字符串内容为大写
    % local -u str=abcABC && echo $str
    ABCABC
    

    设置变量为环境变量

    % local -x str=abc
    # 通常使用 export,功能一样
    % export str=abc
    

    环境变量可以被子进程读取。

    设置变量为只读变量

    % local -r str1=abc
    # 通常使用 readonly,功能一样
    % readonly str2=abc
    
    % str1=bcd
    zsh: read-only variable: str1
    % str2=bcd
    zsh: read-only variable: str2
    

    设置数组不包含重复元素

    % local -U array=(aa bb aa cc) && echo $array
    aa bb cc
    

    设置整数的位数

    # 如果位数不够,输出内容会用 0 补全
    % local -Z 3 i=5 && echo $i
    005
    
    # 如果超出范围会被截断
    % local -Z 3 i=1234 && echo $i
    234
    

    进制转换

    设置整数为其他进制显示:

    % local -i 16 i=255
    % echo $i
    16#FF
    

    可以设置 2 到 36 之间任意进制。设置几进制显示,并不影响计算,只是显示格式不同。

    用 [#n] num 也可以显示十进制数为 n 进制:

    % echo $(([#16] 255))
    16#FF
    

    可以用 n#num 来显示 n 进制整数为十进制:

    % echo $((16#ff))
    255
    

    我们可以定义一系列函数来快捷地转换进制,不需要使用 bc 等外部命令:

    0x() {
        echo $((16#$1))
    }
    
    0o() {
        echo $((8#$1))
    }
    
    0b() {
        echo $((2#$1))
    }
    
    p16() {
        echo $(([#16] $1))
    }
    
    p8() {
        echo $(([#8] $1))
    }
    
    p2() {
        echo $(([#2] $1))
    }
    
    
    # 其他进制转十进制
    % 0x ff
    255
    % 0b 1101
    13
    
    # 十进制转其他进制
    % p16 1234
    16#4D2
    

    同时对多个变量赋相同的值

    % local {i,j,k}=123
    % echo $i $j $k
    123 123 123
    

    绑定字符串和数组

    % local -T DIR dir
    % dir=(/a /b/c /b/d /e/f)
    % echo $DIR
    /a:/b/c:/b/d:/e/f
    
    # 删除 dir 后,DIR 也会被删除(反之亦然)
    % unset dir
    % echo $+DIR
    0
    

    Linux 下经常需要处理带分隔符冒号的字符串(比如 $PATH)。如果只修改其中某一个字段,比较麻烦。local -T 可以把字符串绑定到数组上,这样直接修改数组,字符串内容也会同步变化(反之亦然)。其实在 zsh 中,$PATH 字符串就是和 $path 数组绑定的,可以直接通过修改 $path 来达到修改 $PATH 的目的,这在某些场景会方便很多。

    显示变量的定义方式

    % array=(aa bb cc)
    % local -p array
    typeset -a array=(aa bb cc)
    
    % array+=(dd)
    % local -p array
    typeset -a array=(aa bb cc dd)
    

    什么地方该加双引号

    用过 bash 的读者大概会对里边的双引号印象比较深刻,很多地方不加双引号都会出错,为了避免出错,很多人每个变量左右都加上双引号,麻烦不说,代码看起来也比较乱。

    其实 zsh 中已经没有那些问题了,变量两边无需加双引号,不会出现莫名其妙的错误。但有些地方还是需要加双引号的。

    需要加双引号的场景:

    1. 像这样的包含字符或者特殊符号的字符串 "aa bb \t \n *" 出现在代码中时,两边要加双引号,这个基本不需要说明。
    2. 在用 $() 调用命令时,如果希望结果按一个字符串处理,需要加上双引号,"$()",不然的话,如果命令结果中有空格,$() 会被展开成多个字符串。
    3. 如果想将数组当单个字符串处理,需要加双引号,array=(a b); print -l "$array"
    4. 其他的原本不是单个字符串的东西,需要转成单个字符串的场景,要加双引号。

    其余情况通常都不需要加双引号,典型的情况:

    1. 任何情况下,字符串变量的两边都不需要加双引号,无论里边的内容多么特殊,或者变量存不存在,都没有关系,如 $str
    2. 如果不转换类型(比如数组转成字符串),任何变量的两边都不需要加双引号。
    3. $1 $2 $* 这些参数(其实它们也都是单个字符串),都不需要加双引号,无论内容是什么,或者参数是否存在。

    以上的 7 种情况几乎覆盖了所有场景,如果有没覆盖到的,试一下即可(让里边的内容包含空格、换行和其他特殊字符等等,看看结果是否符合预期)。

    总结

    本文简单介绍了一些比较使用的 typeset(或者 local)命令的用法,typeset 命令还有很多其他参数,但一般很少用,以后我也会继续更新。

    参考

    http://www.bash2zsh.com/zsh_refcard/refcard.pdf

    http://www.linux-mag.com/id/1079/

    更新历史

    20170831:新增“什么地方该加双引号”

    本文不再更新,全系列文章在此更新维护:github.com/goreliu/zshguide

    付费解决 Windows、Linux、Shell、C、C++、AHK、Python、JavaScript、Lua 等领域相关问题,灵活定价,欢迎咨询,微信 ly50247。

    相关文章

      网友评论

        本文标题:Zsh 开发指南(第十一篇 变量的进阶内容)

        本文链接:https://www.haomeiwen.com/subject/migtjxtx.html