Linux基础 -- 环境变量与文件查找

1、环境变量

和多数编程语言中的设置相同,Linux中的bash脚本编程同样可以声明、使用变量。变量声明通过“declare”命令实现,赋值使用“=”(注意不能有空格),取值使用“echo $变量名”。比如:

[root@izwz9gtdx1ch4f9gn56b32z etc]# declare tmp
[root@izwz9gtdx1ch4f9gn56b32z etc]# tmp = pierrecai
-bash: tmp: command not found
[root@izwz9gtdx1ch4f9gn56b32z etc]# tmp=pierrecai
[root@izwz9gtdx1ch4f9gn56b32z etc]# echo $tmp
pierrecai

注意:

  • 和Java中类似,变量名只能是英文字母、数字或者下划线,且不能以数字作为开头。

简单理解了变量的概念,就很容易理解环境变量了。环境变量的作用域比自定义变量的要大,如 Shell 的环境变量作用于自身和它的子进程。在所有的 UNIX 和类 UNIX 系统中,每个进程都有其各自的环境变量设置,且默认情况下,当一个进程被创建时,除了创建过程中明确指定的话,它将继承其父进程的绝大部分环境设置Shell 程序也作为一个进程运行在操作系统之上,而我们在 Shell 中运行的大部分命令都将以 Shell 的子进程的方式运行

通常我们会涉及到的变量类型有三种:

  • 当前 Shell 进程私有用户自定义变量,如上面我们创建的 tmp 变量,只在当前 Shell 中有效
  • Shell 本身内建的变量
  • 从自定义变量导出的环境变量。

也有三个与上述三种环境变量相关的命令:setenvexport。这三个命令很相似,都是用于打印环境变量信息,区别在于涉及的变量范围不同。详见下表:

命 令说 明
set显示当前 Shell 所有变量,包括其内建环境变量(与 Shell 外观等相关),用户自定义变量及导出的环境变量。
env显示与当前用户相关的环境变量,还可以让命令在指定环境中运行。
export显示从 Shell 中导出成环境变量的变量,也能通过它将自定义变量导出为环境变量。

 

关于哪些变量是环境变量,可以简单地理解成在当前进程的子进程有效则为环境变量,否则不是(有些人也将所有变量统称为环境变量,只是以全局环境变量和局部环境变量进行区分,我们只要理解它们的实质区别即可)。

1.1、变量的修改和删除

变量的修改有以下几种方式:

变量设置方式说明
${变量名#匹配字串}从头向后开始匹配,删除符合匹配字串的最短数据
${变量名##匹配字串}从头向后开始匹配,删除符合匹配字串的最长数据
${变量名%匹配字串}从尾向前开始匹配,删除符合匹配字串的最短数据
${变量名%%匹配字串}从尾向前开始匹配,删除符合匹配字串的最长数据
${变量名/旧的字串/新的字串}将符合旧字串的第一个字串替换为新的字串
${变量名//旧的字串/新的字串}将符合旧字串的全部字串替换为新的字串

例如:

[root@izwz9gtdx1ch4f9gn56b32z ~]# path=$PATH
[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $path
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root/mybin
[root@izwz9gtdx1ch4f9gn56b32z ~]# path=${path%/root/mybin}
[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $path
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:

或者还可以使用通配符:

[root@izwz9gtdx1ch4f9gn56b32z ~]# path=$PATH
[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $path
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root/mybin
[root@izwz9gtdx1ch4f9gn56b32z ~]# path=${path%*/mybin}
[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $path
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root

1.2、环境变量删除

可以使用 unset 命令删除一个环境变量

[root@izwz9gtdx1ch4f9gn56b32z ~]# declare tmp
[root@izwz9gtdx1ch4f9gn56b32z ~]# tmp=$PATH
[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $tmp
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root/mybin
[root@izwz9gtdx1ch4f9gn56b32z ~]# unset tmp
[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $tmp

1.3、永久生效

但是问题来了,当你关机后,或者关闭当前的 shell 之后,环境变量就没了啊。怎么才能让环境变量永久生效呢?

按变量的生存周期来划分,Linux 变量可分为两类:

  1. 永久的:需要修改配置文件,变量永久生效;

  2. 临时的:使用 export 命令行声明即可,变量在关闭 shell 时失效。

这里介绍两个重要文件 /etc/bashrc(有的 Linux 没有这个文件) 和 /etc/profile ,它们分别存放的是 shell 变量环境变量。还有要注意区别的是每个用户目录下的一个隐藏文件:

.profile 可以用 ls -a 查看
cd /home/shiyanlou
ls -a

这个 .profile 只对当前用户永久生效。而写在 /etc/profile 里面的是对所有用户永久生效,所以如果想要添加一个永久生效的环境变量,只需要打开 /etc/profile,在最后加上你想添加的环境变量

 

2、 命令的查找路径与顺序

我们在 Shell 中输入一个命令,Shell 是怎么知道去哪找到这个命令然后执行的呢?这是通过环境变量 PATH 来进行搜索的,熟悉 Windows 的用户可能知道 Windows 中的也是有这么一个 PATH 环境变量。这个 PATH 里面就保存了 Shell 中执行的命令的搜索路径。

查看 PATH 环境变量的内容:

[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin

通常这一类目录下(urs/.../(s)bin)放的都是可执行文件,当我们在 Shell 中执行一个命令时,系统就会按照 PATH 中设定的路径按照顺序依次到目录中去查找,如果存在同名的命令,则执行先找到的那个

2.1、一个Bash和C的小例子

分别使用Bash和C编写一个hello小程序:

[root@izwz9gtdx1ch4f9gn56b32z ~]# vim hello_shell.sh
[root@izwz9gtdx1ch4f9gn56b32z ~]# cat hello_shell.sh
#!/bin/bash

for((i=0;i<10;i++));do
	echo "hello shell"
done

exit 0
[root@izwz9gtdx1ch4f9gn56b32z ~]# ls -l
total 4
-rw-r--r-- 1 root root 69 Aug 21 13:08 hello_shell.sh
[root@izwz9gtdx1ch4f9gn56b32z ~]# chmod 755 hello_shell.sh
[root@izwz9gtdx1ch4f9gn56b32z ~]# ls -l
total 4
-rwxr-xr-x 1 root root 69 Aug 21 13:08 hello_shell.sh
[root@izwz9gtdx1ch4f9gn56b32z ~]# ./hello_shell.sh
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
[root@izwz9gtdx1ch4f9gn56b32z ~]# vim hello_world.c
[root@izwz9gtdx1ch4f9gn56b32z ~]# cat hello_world.c 
#include <stdio.h>

int main(void)
{
	printf("hello world!\n");
	return 0;
}
[root@izwz9gtdx1ch4f9gn56b32z ~]# gcc -o hello_world hello_world.c
[root@izwz9gtdx1ch4f9gn56b32z ~]# ls
hello_shell.sh  hello_world  hello_world.c
[root@izwz9gtdx1ch4f9gn56b32z ~]# ./hello_world
hello world!

2.2、添加自定义路径到PATH

注意到,不管是上面创建的可执行的BASH脚本,还是可执行的C程序,我们都必须在文件所在目录才能执行,比如将其移动到另外一个目录里:

[root@izwz9gtdx1ch4f9gn56b32z ~]# mkdir mybin
[root@izwz9gtdx1ch4f9gn56b32z ~]# mv hello_shell.sh hello_world mybin/
[root@izwz9gtdx1ch4f9gn56b32z ~]# ls
hello_world.c  mybin
[root@izwz9gtdx1ch4f9gn56b32z ~]# cd mybin
[root@izwz9gtdx1ch4f9gn56b32z mybin]# ./hello_world 
hello world!
[root@izwz9gtdx1ch4f9gn56b32z mybin]# ./hello_shell.sh 
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
[root@izwz9gtdx1ch4f9gn56b32z mybin]# cd ../
[root@izwz9gtdx1ch4f9gn56b32z ~]# ls
hello_world.c  mybin
[root@izwz9gtdx1ch4f9gn56b32z ~]# ./hello_shell.sh 
bash: ./hello_shell.sh: No such file or directory
[root@izwz9gtdx1ch4f9gn56b32z ~]# ./hello_world 
bash: ./hello_world: No such file or directory

如果希望在任何位置都可以执行命令/脚本文件,我们就需要将对应的目录添加到PATH变量里

注意:

  • 和windows中不同,windows中不同的路径以分号“;”分隔,而在Linux中,不同的路径以冒号“:”分隔。

我们可以这样修改:

[root@izwz9gtdx1ch4f9gn56b32z ~]# PATH=$PATH:/root/mybin
[root@izwz9gtdx1ch4f9gn56b32z ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root/mybin

然后就可以指定在mybin文件夹里的命令:

[root@izwz9gtdx1ch4f9gn56b32z ~]# hello_shell.sh 
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell
hello shell

但是注意:这个修改只在当前Shell有效,一旦我们关闭Shell,这个修改就会失效。

所以我们需要永久地修改PATH,我们可以通过修改初始化环境用的配置脚本来实现永久修改。

每个用户的 home 目录中有一个 Shell 每次启动时会默认执行一个配置脚本,以初始化环境,包括添加一些用户自定义环境变量等等。 Bash 的配置文件为 .bashrc 。在 etc 下还都有一个或多个全局的配置文件,不过我们一般只修改用户目录下的配置文件。

我们可以简单地使用下面命令直接添加内容到 .bashrc中:

[root@izwz9gtdx1ch4f9gn56b32z ~]# echo "PATH=$PATH:/root/mybin" >> .bashrc

上述命令中 >> 表示将标准输出以追加的方式重定向到一个文件中,注意前面用到的 > 是以覆盖的方式重定向到一个文件中,使用的时候一定要注意分辨。在指定文件不存在的情况下都会创建新的文件。

2.3、让环境变量即刻生效

前面我们在 Shell 中修改了一个配置脚本文件之后,每次都要退出终端重新打开甚至重启主机之后其才能生效,很是麻烦,我们可以使用 source 命令来让其立即生效,如:

[root@izwz9gtdx1ch4f9gn56b32z ~]# source .bashrc

source 命令还有一个别名就是 .,注意与表示当前路径的那个点区分开,虽然形式不一样,但作用和使用方式一样,上面的命令如果替换成 . 的方式就该是:

[root@izwz9gtdx1ch4f9gn56b32z ~]# . .bashrc

3、文件搜索

与搜索相关的命令常用的有 whereiswhichfind 和 locate 。

3.1、whereis

whereis 只能搜索二进制文件(-b),man 帮助文件(-m)和源代码文件(-s)。例如:

[root@izwz9gtdx1ch4f9gn56b32z ~]# whereis who
who: /usr/bin/who /usr/share/man/man1/who.1.gz
[root@izwz9gtdx1ch4f9gn56b32z ~]# whereis -m who
who: /usr/share/man/man1/who.1.gz

whereis的搜索速度快,但是能查找的文件类型有限。

3.2、locate

和whereis不同,locate可以用来查找不同类型的文件。

locate通过“ /var/lib/mlocate/mlocate.db ”数据库查找,不过这个数据库也不是实时更新的,系统会使用定时任务每天自动执行 updatedb 命令更新一次,所以有时候你刚添加的文件,它可能会找不到,需要手动执行一次 updatedb 命令(在我们的环境中必须先执行一次该命令)。它可以用来查找指定目录下的不同文件类型,比如查找 /etc 下所有以 sh 开头的文件:

$ locate /etc/sh

注意,它不只是在 /etc 目录下查找,还会自动递归子目录进行查找。

 查找 /usr/share/ 下所有 jpg 文件:

$ locate /usr/share/\*.jpg

注意要添加 * 号前面的反斜杠转义,否则会无法找到。

如果想只统计数目可以加上 -c 参数,-i 参数可以忽略大小写进行查找,whereis 的 -b-m-s 同样可以使用。

centOS7中可能没有内置locate,需要手动进行安装

3.3、which

which 本身是 Shell 内建的一个命令,我们通常使用 which 来确定是否安装了某个指定的软件,因为它只从 PATH 环境变量指定的路径中去搜索命令,比如:

$ which man

3.4、find

 find 应该是这几个命令中最强大的了,它不但可以通过文件类型、文件名进行查找而且可以根据文件的属性(如文件的时间戳,文件的权限等)进行搜索。

这条命令表示去 /etc/ 目录下面 ,搜索名字叫做 interfaces 的文件或者目录。这是 find 命令最常见的格式,千万记住 find 的第一个参数是要搜索的地方:

$ sudo find /etc/ -name interfaces

与时间相关的命令参数:

参数说明
-atime最后访问时间
-ctime最后修改文件内容的时间
-mtime最后修改文件属性的时间

下面以 -mtime 参数举例:

  • -mtime n:n 为数字,表示为在 n 天之前的“一天之内”修改过的文件
  • -mtime +n:列出在 n 天之前(不包含 n 天本身)被修改过的文件
  • -mtime -n:列出在 n 天之内(包含 n 天本身)被修改过的文件
  • -newer file:file 为一个已存在的文件,列出比 file 还要新的文件名

 比如:

列出 home 目录中,当天(24 小时之内)有改动的文件:

$ find ~ -mtime 0

列出用户家目录下比 Code 文件夹新的文件:

$ find ~ -newer /home/shiyanlou/Code

转载于:https://my.oschina.net/pierrecai/blog/1518338

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值