PS : 注意本文讨论的是Bash,而不一定是/bin/sh所链接的那个shell。这里出现的所有代码片段,默认在顶上都添加了#!/bin/bash。

一门自带混淆的语言

while (( $# )); do

case $1 in

-a*)

# Error checking

[[ ${1#-a} ]] || { echo "bash: ${FUNCNAME[0]}: \`$1': missing"\

"number specifier" 1>&2; return 1; }

printf %d "${1#-a}" &> /dev/null || { echo "bash:"\

"${FUNCNAME[0]}: \`$1': invalid number specifier" 1>&2

return 1; }

# Assign array of -aN elements

[[ "$2" ]] && unset -v "$2" && eval $2=\(\"\${@:3:${1#-a}}\"\) &&

shift $((${1#-a} + 2)) || { echo "bash: ${FUNCNAME[0]}:"\

"\`$1${2+ }$2': missing argument(s)" 1>&2; return 1; }

;;

-v)

# Assign single value

[[ "$2" ]] && unset -v "$2" && eval $2=\"\$3\" &&

shift 3 || { echo "bash: ${FUNCNAME[0]}: $1: missing"\

"argument(s)" 1>&2; return 1; }

;;

*)

echo "bash: ${FUNCNAME[0]}: $1: invalid option" 1>&2

return 1 ;;

esac

done

如果你觉得阅读上面的Bash代码,就像阅读段子一样顺畅,那么是时候关掉这个页面,去做别的更有意义的行为,比如去喝个水什么的。
如果你觉得上面的Bash代码犹如鬼画符,并且实际生活中不得不面对它,那么就看下去吧。

Bash变量操作

正式开始正文内容。

想要在一篇文章里,讲述要看懂开篇代码所需的所有知识点,这是不自量力的行为。因此,本文将讲且仅讲Bash中操作变量的方法。所以,即使你看完了这篇文章,你多半还是看不懂开篇的代码。

不过看完这篇文章之后,你对Bash的变量操作会有更为深入的认识。而且更重要的是,Bash之于你,不再是怎么也看不清摸不透。下一次要写脚本的时候,你也将更加坚定地下定决心 —— 人生苦短,我用Python/Ruby。

严格意义上的Bash变量类型

Bash变量只有两种类型,字符串和数组。不过从严格意义上,Bash没有变量类型。Bash中的变量,在运行的时候会被展开成其对应的值(字符串)。你可以把它看做C/C++中的宏定义,或者一些模板语言中的占位符。

一般情况下,变量通过=赋值,注意=两边不要留空格。有些好孩子,已经养成了符号两端留空格的习惯,结果当开始写Bash的时候,他们抓狂了。

要想访问变量,只需在变量名前面添加$,解释器就会对它进行展开。如果该变量并不存在,解释器会把它展开成“”。

me=spacewander

echo $me

echo $who

来自命令行的你

作为脚本语言,第一要义当然是要随时随地获取到命令行输入啦。

在Bash中,使用$1可以获取命令行输入的第一个参数,$2可以获取命令行输入的第2个参数,$3可以获取命令行输入的第......

你看,$1到$10000的用法就这么交代完了。Bash还是挺有逻辑的嘛。

顺便一提,$0获取的脚本的名字(其实就是其他语言中的第0个参数),$@获取所有的参数,$#获取参数的数目。记住@和#这两个符号,在Bash这一神秘的符文体系中,前者表示全部参数,后者表示参数的数目。

展开,然后Bomb!

假如Bash变量中含有空白字符,或者含有特殊字符,比如*,展开后会污染到外面的字符串,结果就是Bomb。

比如

Oops='*'

# '*'解释成所有匹配的文件名

echo $Oops

# 所以需要加双引号括起来

echo "$Oops"

# 加单引号会怎样呢?

echo '$Oops'

上面的代码值得一试。

另外一种Bomb的可能是,变量后面需要接其它字符串,比如$FRUITs。如果想让解释器识别成$FRUIT变量,而不是$FRUITs,需要用花括号括起来,像${FRUIT}s

数组和关联数组

Bash中可以使用两种容器。

一种是数组,另一种是关联数组,类似于其他语言中的Map/Hash/Dict。

声明数组的常用语法: declare -a ARY或者ARY=(1 2 3)

声明关联数组的唯一语法: declare -A MAP

赋值的语法:

直接ARY[N]=VALUE,N可以是数字索引也可以是键。关联数组可以使用MAP=([x]=a [y]=b)进行多项赋值,注意这是赋值的语句而不是声明。

亲测数组中的索引不一定要按顺序来,你可以先给2和3上的元素赋值。(同样算是弱类型的Javascript也支持这种无厘头赋值,这算通病么?)

往现有数组批量添加元素:

ARY+=(a b c)

MAP+=([a]=1 [b]=2)

取值:

${ARY[INDEX]}

${MAP[KEY]}

注意花括号的使用

${A[@]} 展开成所有的变量,而获取数组长度使用 ${#A[@]}

切片:

${ARY[@]:N:M} N是offset而M是length

返回索引,相当于keys():

${!MAP[@]}

试试下面的代码:

declare -a ARY

declare -A MAP

MAP+=([a]=1 [b]=2)

ARY+=(a b c)

echo ${ARY[1]}

echo ${MAP[a]}

echo "${ARY[@]}"

echo "${MAP[@]}"

echo "${ARY[@]:0:1}"

echo ${#ARY[@]}

echo "${!MAP[@]}"

ARY[4]=a

echo ${ARY[@]}

echo ${ARY[3]}

变量(字符串)变换

Bash中的变量变换,大体是${变量[操作符]}的形式

大小写变换

HI=HellO

echo "$HI" # HellO

echo ${HI^} # HellO

echo ${HI^^} # HELLO

echo ${HI,} # hellO

echo ${HI,,} # hello

echo ${HI~} # hellO

echo ${HI~~} #hELLo

^大写,,小写, ~大小写切换

重复一次只修改首字母,重复两次则应用于所有字母。

混着用会怎样?

echo ${HI^,^} # HellO

看来是不行的×_×

移除匹配的字符串

%xx 从后往前,开始匹配,移除匹配的内容

%%xx 跟上面的差不多,不过这是贪婪匹配
#xx 从前往后,开始匹配,移除匹配的内容
##xx 跟上面的差不多,不过这是贪婪匹配

这个比较难理解,不过看下面几个例子应该能明白了。

FILENAME=/home/spacewander/param.sh

echo ${FILENAME%/*} # /home/spacewander

echo ${FILENAME%%/*} #

echo ${FILENAME#*/} # home/spacewander/param.sh

echo ${FILENAME##*/} # param.sh

查找并替换

/MATCH/VALUE 替换第一个匹配的内容。

//MATCH/VALUE 替换匹配的内容

echo ${FILENAME/home/office} # /office/spacewander/param.shecho ${FILENAME//s/S} # /home/Spacewander/param.Sh

其它字符串操作

获取变量(字符串)长度:${#FILENAME}

字符串切片:跟数组切片是同样的语法,${STR:offset:len}

TEXT=这个程序充满了BUG!

echo ${TEXT:0:8}

echo ${TEXT:4}

# 你还可以使用负数作为offset,这时候就是从后往前算起。注意负数要用括号括起来,避免冲突。

echo ${TEXT:(-4)}

关于变量,其它的内容

Bash中有一项特性,你可以方便地检查某个变量是否设置了,如果没有设置,就赋予一个默认值。尤其在处理环境变量的时候,这项特性会让你感到欣慰。

语法是${VAR:=VALUE}或者${VAR:=VALUE}。此外,还有一个相似的语法,${VAR:=VALUE}和${VAR:=VALUE}。

下面展示下两者的区别

# expand to default variable

echo ${NULL-"Not null"} # Not null

echo ${NULL} #

# set default variable

echo ${NIL="Not nil"} # Not nil

echo ${NIL} # Not nil

可以看出,前者只是当变量不存在时,展开成指定的值。而后者在变量不存在时,将变量的值设置为指定值。

最后介绍一个,当目标变量不存在时,指定报错信息的语法。

echo ${TARGET?Not Found} # 当$TARGET不存在时,显示TARGET: Not Found,并结束程序。

转 玩转Bash变量的更多相关文章

  1. 玩转Bash脚本:test測试语句

    总第1篇test就是測试的意思,经常使用在流程控制语句中作为条件.以下做一下介绍. 关于真值 与其它语言不同,Bash(包含其它Shell)中,是用0表示真,非0表示假的.之所以用0表示成功,而不是1 ...

  2. 玩转Bash脚本:循环结构之while循环(转)

    转自:http://blog.csdn.net/guodongxiaren/article/details/43341769   总第8篇 本系列(玩转Bash脚本)更多文章,请访问:http://b ...

  3. 玩转Bash脚本:选择结构之case

    总第5篇 之前,我们谈到了if. 这次我们来谈还有一种选择结构--case. case与if if用于选择的条件,不是非常多的情况,假设选择的条件太多.一系列的if.elif,.也是醉了. 没错,ca ...

  4. Shell基础-Bash变量-用户自定义变量

    变量设置规则: 变量名称可以由字母.下划线和数字组成,但是不能由数字开头. 在Bash中变量的默认类型是字符串类型,若需要进行数值运算,则需指定变量类型为数值型.变量用等号链接,且两边不能有空格.若需 ...

  5. Shell学习之Bash变量详解(二)

    Shell学习之Bash变量详解 目录 Bash变量 Bash变量注意点 用户自定义变量 环境变量 位置参数变量 预定义变量 Bash变量 用户自定义变量:在Bash中由用户定义的变量. 环境变量:这 ...

  6. linux bash变量作用域

    linux bash变量作用域 一,思考一个问题,当在shell里执行某个程序时,shell是怎么找到这个程序的? shell会去$PATH环境变量定义的目录里去找这个命令.环境变量里一般包括/usr ...

  7. linux学习18 shell脚本基础-bash变量和逻辑运行

    一.回顾 1.用户管理,权限管理,install,mktemp 2.用户管理: 3.权限管理: mode,ownership mode: user group other r w x 4.命令:ins ...

  8. 二:shell之bash变量

    1.变量的分类: 用户自定义变量:   变量自定义 默认存储是字符串环境变量:              这种变量中主要保存的是和系统操作环境相关的数据.变量可以自定义,但是对系统生效的环境变量名和变 ...

  9. Linux学习笔记(16)shell基础之Bash变量

    1. 用户自定义变量 (1)变量设置规则 ① 变量名称可由字母.数字和下划线组成,但不能以数字开头: ② 变量的默认类型为字符串类型,如果要对数值运算,则必须指定变量类型为数值型: ③ 变量用等号连接 ...

随机推荐

  1. poi的各种单元格样式以及一些常用的配置

    之前我做过一个poi到处excel数据的博客,但是,后面使用起来发现,导出的数据单元格样式都不对. 很多没有居中对齐,很多单元格的格式不对,还有就是单元格的大小不对,导致数据显示异常,虽然功能可以使用 ...

  2. hdu_3709_Balanced Number(数位DP)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=3709 题意:给你一个区间,让你找平衡数的个数 题解:设dp[i][j][k]为前i位以第j位为支撑点的 ...

  3. Entity Framework教程

    随着Code First一起出现的DbContext和DbSet类绝对可以称得上EF的功能核心,其取代了之前的ObjectContext和ObjectSet类,提供了与数据库通信,管理内存中实体的重要 ...

  4. iOS:UIWebView scrollView 的分页滑动问题

    最近在弄一个native webview+html的项目,感觉这种尝试还是挺不错的,特被是适合内容类app.如杂志.电子书等.其实native搭的就是一个框架,主体还是在html的内容上,所以花在ht ...

  5. word转pdf swf 在线预览

    来源:http://www.cnblogs.com/wuhenke/archive/2010/08/01/1789750.html 之前在项目中研究使用了一套word转PDF,然后将PDF转成SWF的 ...

  6. Docker私有仓库Registry 搭建

    1. 关于Registry 官方的Docker hub是一个用于管理公共镜像的好地方,我们可以在上面找到我们想要的镜像,也可以把我们自己的镜像推送上去.但是,有时候,我们的使用场景需要我们拥有一个私有 ...

  7. PHP运行模式(cgi,fast-cgi,cli, ISAPI ,web模块模式)【转载】

    PHP运行模式有5钟: 1)cgi 通用网关接口(Common Gateway Interface))2)fast-cgi 常驻 (long-live) 型的 CGI3)cli  命令行运行   (C ...

  8. perl的Getopt::Long和pod::usage ?

    来源: http://www.cnblogs.com/itech/archive/2012/08/07/2627267.html 代码: 需要显式地定义变量且初始化.例如optionX. 如果没有定义 ...

  9. asp之servervariables全部显示

    <%Response.Write("<p>")%><%for each i in request.servervariables%> <% ...

  10. wpf之数据触发器DataTrigger

    wpf, 根据绑定的属性的值的不同(数据分类),界面上显示不同的控件(绑定不同类型的属性),可以使用数据库触发器DataTrigger实现这一功能. 实现的效果如下: 首先建立实体类: 更改通知类: ...