001. 变量、环境变量

1、在终端中显示输出

shell脚本通常以shebang起始:#!/bin/bash/

  • shebang是一个文本行,其中#!位于解释器路径之前。
  • /bin/bash是Bash的解释器命令路径。
  • bash将以#符号开头的行视为注释。脚本中只有第一行可以使用shebang来定义解释该脚本所使用的解释器。

脚本的执行方式有两种:

  • 第一种,将脚本名作为命令行参数:bash myScript.sh如果将脚本作为bash的命令行参数来运行,那么就用不着使用shebang了。

  • 第二种,授予脚本执行权限,将其变为可执行文件。可以利用shebang来实现脚本的独立运行。可执行脚本使用shebang之后的解释器路径来解释脚本。

    赋予脚本可执行权限

    chmod 755 myScript.sh
    chmod a+x myScript.sh

    执行脚本

    ./myScript.sh # ./表示当前目录
    /home/path/myScript.sh # 使用脚本的完整路径

    内核会读取脚本的首行并注意到shebang为#!/bin/bash。它会识别出/bin/bash并执行该脚本:

    /bin/bash myScript.sh

当启动一个交互式shell时,它会执行一组命令来初始化提示文本、颜色等设置。这组命令来自用户主目录中的脚本文件~/.bashrc(对于登录shell则是~/.bash_profile)。Bash shell还维护了一个历史记录文件~/.bash_history,用于保存用户运行过的命令。

**注意:**登录shell是登录主机后创建的那个shell。但登录图形化环境(比如GNOME、KDE等)后所创建的终端会话并不是登录shell。使用GNOME或KDE这类显示管理器登录后并不会读取.profile或.bash_profile(绝大部分情况下不会),而使用ssh登录远程系统时则会读取.profile。shell使用分号或换行符来分隔单个命令或命令序列。

a. echo 命令
  • 描述:echo是用于终端打印的最基本命令。默认情况下,echo在每次调用后会添加一个换行符。
  • 语法:echo [选项] "内容"
    • -e 包含转义序列的字符串

    • -n 禁止在文本的尾部追加一个换行符

      echo Welcome to bash
      echo 'Welcome to bash'
      echo "Weclome to bash"

      如果不使用引号,我们无法在echo中使用分号,因为分号在Bash shell中用作命令间的分隔符:

      echo hello;echo xixi
      hello
      xixi

  • 这些方法看起来相似,但各有特定的用途及副作用。双引号允许shell解释字符串中出现的特殊字符。单引号不会对其做任何解释。

转义字符:

默认情况下,echo会在输出文本的尾部追加一个换行符。可以使用选项-n来禁止这种行为。echo同样接受双包含转义序列的双引号字符串作为参数。在使用转义序列时,需要使用echo -e "包含转义序列的字符串"这种形式。

复制代码
echo -e "1\t2\t3"				# 输出:1	2	3 而不是 1\t2\t3
echo -n haha						# 输出时文本尾部没有换行

打印彩色输出:

脚本可以使用转义序列在终端中生成彩色文本:

  • 文本颜色是由对应的色彩码来描述的。其中包括:重置=0,黑色=30,红色=31,绿色=32,黄色=33,蓝色=34,洋红=35,青色=36,白色=37。

  • 对于彩色背景,经常使用的颜色码是:重置=0,黑色=40,红色=41,绿色=42,黄色=43,蓝色=44,洋红=45,青色=46,白色=47。

    打印彩色文本

    echo -e "\e[1;31m This is red text \e[0m" # 以红色字体打印 This is red text

    设置彩色背景

    echo -e "\e[1;42m Green Background \e[0m" # 将 Green Bachground 的文本背景设置为绿色

  • 其中\e[1;31m是一个转义字符串,可以将颜色设为红色,\e[0m将颜色重新置回。

  • 这些例子中包含了一些转义序列。可以使用man console_codes来查看相关文档。

b. prinf 命令

printf命令接受引用文本或由空格分隔的参数。默认情况下,printf并不会自动添加换行符,需手动指定。

可以在printf中使用格式化字符串来指定字符串的宽度、左右对齐方式等。

  • 描述:用于终端打印

  • 语法:printf "内容"

    #!/bin/bash

    printf "%-5s %-10s %-4s\n" No Name Mark
    printf "%-5s %-10s %-4.2f\n" 1 Sarath 80.3456
    printf "%-5s %-10s %-4.2f\n" 2 James 90.9989
    printf "%-5s %-10s %-4.2f\n" 3 Jeff 77.564

  • %s、%c、%d和%f都是格式替换符(format substitution character),它们定义了该如何打印后续参数。
    • %-5s指明了一个格式为左对齐且宽度为5的字符串替换(-表示左对齐)。如果不指明-,字符串就采用右对齐形式。
    • 宽度指定了保留给某个字符串的字符数量。对Name而言,其保留宽度是10。因此,任何Name字段的内容都会被显示在10字符宽的保留区域内,如果内容不足10个字符,余下的则以空格填充。
  • 对于%f,可以使用其他参数对小数部分进行舍入(round off)。对于Mark字段,我们将其格式化为%-4.2f,其中.2指定保留两位小数。注意,在每行的格式字符串后都有一个换行符(\n)。
2、使用变量和环境变量

所有的编程语言都利用变量来存放数据,以备随后使用或修改。和编译型语言不同,大多数脚本语言不要求在创建变量之前声明其类型。用到什么类型就是什么类型。在变量名前面加上一个美元符号就可以访问到变量的值。shell定义了一些变量,用于保存用到的配置信息,比如可用的打印机、搜索路径等。这些变量叫作环境变量。

变量名由一系列字母、数字和下划线组成,其中不包含空白字符。常用的惯例是在脚本中使用大写字母命名环境变量,使用驼峰命名法或小写字母命名其他变量。

a. env | printenv 命令
  • 描述:查看当前shell中所定义的全部环境变量。
  • 语法:env | printenv
b. pgrep 命令
  • 描述:查看相关进程的进程ID

  • 语法:pgrep 进程名称

    假设有一个叫作gedit的应用程序正在运行。我们可以使用pgrep命令获得gedit的进程ID:

    pgrep gedit

    假设查到的 gedit 的进程ID为 12501,那么可以通过如下命令查看与该进程相关的环境变量:

    cat /proc/12501/environ

    要想生成一份易读的报表,可以将cat命令的输出通过管道传给tr,将其中的\0替换成\n:

    cat /proc/12501/environ | tr '\0' '\n'

  • 特殊文件/proc/PID/environ是一个包含环境变量以及对应变量值的列表。每一个变量以name=value的形式来描述,彼此之间由null字符(\0)分隔。

c. 变量赋值

使用等号操作符为变量赋值:varName=value

  • 如果value不包含任何空白字符(例如空格),那么就不需要将其放入引号中,否则必须使用单引号或双引号。
  • 注意:var = value不同于 var=value。把var=value写成var = value是一个常见的错误。两边没有空格的等号是赋值操作符,加上空格的等号表示的是等量关系测试。

在变量名之前加上美元符号($)就可以访问变量的内容。可以在printf、echo或其他命令的双引号中引用变量值:

复制代码
#!/bin/bash

fruit=apple
count=5
echo "We have $count ${fruit}(s)"

d. 环境变量

环境变量是从父进程中继承而来的变量。例如环境变量HTTP_PROXY,它定义了Internet连接应该使用哪个代理服务器。该环境变量通常被设置成:

复制代码
HTTP_PROXY=192.168.1.23:3128
export HTTP_PROX
  • export命令声明了将由子进程所继承的一个或多个变量。
  • 这些变量被导出后,当前shell脚本所执行的任何应用程序都会获得这个变量。shell创建并用到了很多标准环境变量,我们也可以导出自己的环境变量。

PATH变量列出了一系列可供shell搜索特定应用程序的目录:

复制代码
echo $PATH
	-> /usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
  • 各目录路径之间以:分隔。$PATH通常定义在/etc/environment、/etc/profile或~/.bashrc中。

    cat /etc/profile

如果需要在PATH中添加一条新路径,可以使用如下命令:

复制代码
export PATH="$PATH:/home/user/bin"		# 在环境变量中添加新的路径
echo $PATH
 -> /usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/home/usr/bin

另外还有一些众所周知的环境变量:HOME、PWD、USER、UID、SHELL等。

**注意:**使用单引号时,变量不会被扩展,仍依照原样显示。使用双引号会显示出该变量的值

复制代码
echo '$PATH'
	-> $PATH
  
echo "$PATH"
	-> /usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/home/usr/bin: No such file or directory

e. 获取变量值的长度
复制代码
$ var=12345678901234567890
$ echo ${#var}
20
f. 识别当前使用的Shell
复制代码
echo $SHELL
echo $0
3、使用函数添加环境变量

环境变量通常保存了可用于搜索可执行文件、库文件等的路径列表,如$PATH。

复制代码
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin

这意味着只要shell执行应用程序(二进制文件或脚本)时,它就会首先查找/usr/local/sbin,然后查找usr/local/bin 依次查找。

使用源代码构建并安装程序时,通常需要为新的可执行文件和库文件添加特定的路径。假设我们要将myapp安装到/opt/myapp,它的二进制文件在/opt/myapp/bin目录中,库文件在/opt/myapp /lib目录中。

将新路径添加到环境变量起始部分:

复制代码
export PATH=/opt/myapp/bin:$PATH

使用函数添加环境变量:( 案例没做出来

我们可以在.bashrc文件中定义如下函数,简化路径添加操作:

复制代码
prepend() { [ -d "$2" ] && eval $1=\"$2':'\$$1\" && export $1; }

该函数用法如下:

复制代码
prepend PATH /opt/myapp/bin
  • 函数prepend()首先确认该函数第二个参数所指定的目录是否存在。
  • 如果存在,eval表达式将第一个参数所指定的变量值设置成第二个参数的值加上:(路径分隔符),随后再跟上第一个参数的原始值。

在进行添加时,如果变量为空,则会在末尾留下一个:。要解决这个问题,可以对该函数再做一些修改:

复制代码
prepend() { [ -d "$2" ] && eval $1=\"$2\$\{$1:+':'\$$1\}\" && export $1 ; }
  • 在这个函数中,我们引入了一种shell参数扩展的形式:${parameter:+expression}
  • 如果parameter有值且不为空,则使用expression的值。
  • 通过这次修改,在向环境变量中添加新路径时,当且仅当旧值存在,才会增加:。

相关推荐
dingdingfish3 天前
GNU Parallel 学习 - 第1章:How to read this book
bash·shell·gnu·parallel
似霰5 天前
Linux Shell 脚本编程——核心基础语法
linux·shell
似霰6 天前
Linux Shell 脚本编程——脚本自动化基础
linux·自动化·shell
偷学技术的梁胖胖yo7 天前
Shell脚本中连接数据库查询数据报错 “No such file or directory“以及函数传参数组
linux·mysql·shell
纵有疾風起16 天前
【Linux 系统开发】基础开发工具详解:软件包管理器、编辑器。编译器开发实战
linux·服务器·开发语言·经验分享·bash·shell
gis分享者18 天前
Shell 脚本中如何使用 here document 实现多行文本输入? (中等)
shell·脚本·document·多行·文本输入·here
柏木乃一18 天前
基础IO(上)
linux·服务器·c语言·c++·shell
angushine19 天前
CPU脚本并远程部署
shell
赵民勇23 天前
Linux/Unix中install命令全面用法解析
linux·shell
gis分享者24 天前
Shell 脚本中如何使用 trap 命令捕捉和处理信号(中等)
shell·脚本·信号·处理·trap·捕捉