美文网首页诗翔的Linux学习之路生物信息学与算法我用 Linux
【shell笔记>命令|变量】理解内建命令与环境变量

【shell笔记>命令|变量】理解内建命令与环境变量

作者: 王诗翔 | 来源:发表于2017-06-29 16:14 被阅读148次

    2017.6.29

    理解shell内建命令

    理解内建命令和外部命令非常重要。相比之下,外部命令使用成本更高。

    外部命令

    外部命令,是存在于bash shell之外的程序,不是shell程序的一部分。通常位于/bin, /usr/bin, /sbin, /usr/sbin中。

    例如ps,我们来看看

    wsx@wsx-ubuntu:~$ which ps
    /bin/ps
    wsx@wsx-ubuntu:~$ type -a ps
    ps 是 /bin/ps
    wsx@wsx-ubuntu:~$ ls -l /bin/ps
    -rwxr-xr-x 1 root root 97408 11月 22  2016 /bin/ps
    
    

    当外部命令执行时,会创建出一个子进程。这种操作成为衍生(forking)

    内建命令

    内建命令和外部命令的区别在于前者不需要使用子进程来执行。它们已经和shell编译成了一体,作为shell工具的组成部分存在。不需要借助外部程序文件来运行。

    例如cdexit都内建于bash shell。可以利用type命令来了解某个命令是否是内建的。

    wsx@wsx-ubuntu:~$ type cd
    cd 是 shell 内建
    wsx@wsx-ubuntu:~$ type exit
    exit 是 shell 内建
    

    有意思的是,有些命令有多种实现方式,即它既有内建,也有外部命令方式。可以通过type -a查看。

    wsx@wsx-ubuntu:~$ type -a echo
    echo 是 shell 内建
    echo 是 /bin/echo
    wsx@wsx-ubuntu:~$ which echo
    /bin/echo
    wsx@wsx-ubuntu:~$ type -a pwd
    pwd 是 shell 内建
    pwd 是 /bin/pwd
    wsx@wsx-ubuntu:~$ which pwd
    /bin/pwd
    
    

    可以看到,type -a命令显示了每个命令的两种实现。而which命令只显示出了外部命令文件。

    使用history命令

    bash shell会跟踪你用过的命令,我们可以唤回这些命令并重新使用。通常记录保存1000条,可以通过HISTSIZE环境变量进行修改。

    通过!!就可以重新唤出刚刚用过的命令

    wsx@wsx-ubuntu:~$ !!
    which pwd
    /bin/pwd
    

    当我们使用多个终端会话进行工作时,历史记录并不会自动更新,可以通过history -n命令强制重新读入。

    环境变量

    bash shell用一种叫做环境变量的特性来存储有关shell会话和工作环境的信息。这项特性允许你在内存中存储数据,以便程序或shell中运行的脚本能够轻松访问到它们。

    分为两类:

    • 全局变量
    • 局部变量

    全局环境变量

    全局环境变量对于shell会话和所有生成子shell都是可见的。局部变量则只对创建它们的shell可见。

    Linux系统在你开始bash会话时就设置了一些全局环境变量。系统环境变量基本上都是使用全大写字母,以区别于普通用户的环境变量。

    envprintenv命令可以查看全局变量:

    wsx@wsx-ubuntu:~$ env
    XDG_VTNR=7
    XDG_SESSION_ID=c2
    CLUTTER_IM_MODULE=xim
    XDG_GREETER_DATA_DIR=/var/lib/lightdm-data/wsx
    GPG_AGENT_INFO=/home/wsx/.gnupg/S.gpg-agent:0:1
    TERM=xterm-256color
    SHELL=/bin/bash
    VTE_VERSION=4205
    QT_LINUX_ACCESSIBILITY_ALWAYS_ON=1
    WINDOWID=102760458
    UPSTART_SESSION=unix:abstract=/com/ubuntu/upstart-session/1000/1819
    GNOME_KEYRING_CONTROL=
    GTK_MODULES=gail:atk-bridge:unity-gtk-module
    JRE_HOME=/usr/lib/jvm/default-java/jre
    ......
    

    环境变量很多,不大好找。如果想要显示个别环境变量的值,可以用printenv,不要用env命令。

    wsx@wsx-ubuntu:~$ printenv HOME
    /home/wsx
    

    可以用echo命令显示变量的值

    wsx@wsx-ubuntu:~$ echo $HOME
    /home/wsx
    

    加上$的变量还可以作为命令的参数

    wsx@wsx-ubuntu:~$ ll $HOME
    总用量 236220
    drwxr-xr-x  65 wsx  wsx       4096 6月  29 14:19 ./
    drwxr-xr-x   3 root root      4096 9月  25  2016 ../
    drwx------   3 wsx  wsx       4096 9月  25  2016 .adobe/
    drwxrwxr-x   3 wsx  wsx       4096 4月  13 10:59 .anaconda/
    drwxrwxr-x   4 wsx  wsx       4096 4月  14 15:05 .anydesk/
    -rw-------   1 wsx  wsx      54981 6月  29 10:50 .bash_history
    -rw-r--r--   1 wsx  wsx        220 9月  25  2016 .bash_logout
    -rw-r--r--   1 wsx  wsx       4203 6月  19 18:39 .bashrc
    -rw-r--r--   1 wsx  wsx       4122 6月  19 18:39 .bashrc-anaconda3.bak
    ......
    

    局部变量

    顾名思义,局部环境变量只能在定义它们的进程中可见。这两点的区分相信接触过编程的朋友都很容易理解,因为在一般编程语言函数内外,我们也必须涉及到这个概念的掌握。进程或者子shell,我认为都可以看做某种函数。

    事实上,Linux系统也默认定义了标准的局部环境变量。我们也可以自定义。

    颇为遗憾的是,在Linux系统上并没有一个只显示局部环境变量的命令。set命令会显示为某个特定进程设置的所有环境变量。

    设置用户自定义变量

    设置局部用户自定义变量

    wsx@wsx-ubuntu:~$ echo $my_variable # 因为没定义,可以看到输出为空
    
    wsx@wsx-ubuntu:~$ my_variable=Hello # 等号两边不要留空格!
    wsx@wsx-ubuntu:~$ echo $my_variable 
    Hello
    wsx@wsx-ubuntu:~$ my_variable="Hello World" # 如果字符串还有空格,用引号括起来
    wsx@wsx-ubuntu:~$ echo $my_variable
    Hello World
    wsx@wsx-ubuntu:~$ bash     
    wsx@wsx-ubuntu:~$ echo $my_variable       # 在子shell中无法使用用户自定义变量
    
    

    设置全局环境变量

    创建的方式是先创建一个局部环境变量,然后再把它导出到全局环境中。这个过程用export命令来完成(但是退出终端后会失效,需要保存到配置文件才能永久生效)。

    删除环境变量

    unset可以完成这个操作,记住不要用$

    记住一点:只要用到变量,使用$;如果要操作变量,不使用$。一个例外是使用printenv显示某个变量的值。

    设置PATH环境变量

    当我们在shell命令行界面输入一个外部命令时,shell必须搜索系统来找到对应的程序。PATH环境变量定义了用于进行命令和程序寻找的目录。

    wsx@wsx-ubuntu:~$ echo $PATH
    /home/wsx/Anaconda/bin:/home/wsx/Anoconda/ENTER/bin:/usr/lib/jvm/default-java/bin:/home/wsx/bin:/home/wsx/Anaconda/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin
    

    目录使用冒号分隔。

    可以添加搜索目录到现有的PATH环境变量,无需从头定义。你只需引用原来的PATH值,然后再给这个字符串添加新目录就行了。

    $ PATH=$PATH:/home/scripts
    $ echo $PATH
    

    程序员通常的办法是将单点符也加入PATH环境变量。该单点符代表当前目录。

    $ PATH=$PATH:.
    

    要记住,由父shell设置但并未导出的变量都是局部变量,子shell无法继承局部变量。

    数组变量

    要给某个环境变量设置多个值,可以把值放在括号里,值与值之间用空格分隔。

    wsx@wsx-ubuntu:~$ mytest=(one two three four five)
    wsx@wsx-ubuntu:~$ echo $mytest
    one
    wsx@wsx-ubuntu:~$ echo ${mytest[2]}  # 索引值从0开始
    three
    wsx@wsx-ubuntu:~$ echo ${mytest[*]}
    one two three four five
    wsx@wsx-ubuntu:~$ mytest[2]=seven # 改变某个索引值位置的值
    wsx@wsx-ubuntu:~$ echo ${mytest[*]}
    one two seven four five
    wsx@wsx-ubuntu:~$ unset mytest[2]  # 删除某个变量
    wsx@wsx-ubuntu:~$ echo ${mytest[*]}
    one two four five
    
    

    相关文章

      网友评论

        本文标题:【shell笔记>命令|变量】理解内建命令与环境变量

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