Shell编程基础及变量

一、Shell脚本

1.Shell脚本的建立

由Linux命令、shell命令、程序结构控制语句和注释等内容组成。

脚本第一行

#!/bin/bash
#!字符称为幻数,内核会根据它后面的解释器来确定该用什么程序解释脚本

bash与sh的区别
sh为bash的软连接,多数情况下无区别

[xiaoda@codis-178 ~]$ ll /bin/sh
lrwxrwxrwx. 1 root root 4 Mar  4  2016 /bin/sh -> bash
[xiaoda@codis-178 ~]$ ll /bin/bash
-rwxr-xr-x. 1 root root 1009968 Nov 17  2014 /bin/bash

检查bash是否有漏洞

[root@codis-178 ~]# env x='() { :;}; echo be careful' bash -c "echo this is a test"
this is a test

如果返回以下两行,则尽快升级
be careful
this is a test

若不指定解释器,则用对应解释器执行脚本
bash test.sh
python test.py
expect test.exp

2.脚本注释

解释器看不到,不会执行
可跟在脚本命令的后面,与命令在同一行
尽量不要用中文

3.shell脚本的执行


(1)查找系统环境变量ENV,该变量指定了环境文件,加载顺序/etc/profile ->.bash_profile->.bashrc->/etc/bashrc
(2)执行脚本
(3)自上而下,从左到右依次执行(脚本嵌套时,先执行自脚本)

4.执行脚本的方式

(1)bash script-name 或 sh script-name
(2)path/script-name 或 ./script-name
(3)source script-name 或 . script-name
(4)sh<script-name 或 cat script-name|sh

示例:

已知以下命令及返回结果,请问echo $user的返回结果是
cat test.sh
user='whoami'
sh test.sh
echo $user

a.当前用户
b.oldboy
c.无内容输出

答案是:c
解释:
sh和bash执行脚本会启动新的子shell,执行完后退回到父shell,因此变量或函数值无法保留。

结论:

  • 子shell脚本会直接继承父shell脚本的变量、函数等,反之不可
  • 若想反过来继承,要用source 或 "." 在父shell中事先加载子shell

二、变量

1.变量特性

shell中不区分变量的类型(整数、小数、字符串等)

2.变量类型

(1)环境变量(全局)
export内置命令导出的变量(用户名、路径、目录)
名字采用大写形式
set命令输出所有变量(全局和局部)
env命令只显示全局

$HOME   #用户家目录
$UID    #当前用户的UID
$PWD    #当前工作目录的绝对路径
$SHELL  #当前shell
$USER   #当前用户

环境变量初始化及生效顺序

  • 登录系统,加载主环境变量文件/etc/profile
  • 执行/etc/profile.d目录下的脚本
  • 运行$HOME/.bash_profile用户环境变量文件
  • 在用户环境变量文件中会去找$HOME/.bashrc,如果有这个文件则执行,没有找/etc/bashrc全局环境变量文件

(2)普通变量(局部)
由字母、数字、下划线组成
可以用单引号或双引号引起来

[xiaoda@codis-178 ~]$ a=192.168.16.1
[xiaoda@codis-178 ~]$ b='192.168.16.1'
[xiaoda@codis-178 ~]$ c="192.168.16.1"
[xiaoda@codis-178 ~]$ echo "a=$a"
a=192.168.16.1
[xiaoda@codis-178 ~]$ echo "b=$b"
b=192.168.16.1
[xiaoda@codis-178 ~]$ echo "c=$c"
c=192.168.16.1
[xiaoda@codis-178 ~]$ echo "c=${c}"
c=192.168.16.1
[xiaoda@codis-178 ~]$ a=192.168.16.1-$a
[xiaoda@codis-178 ~]$ b='192.168.16.1-$a'
[xiaoda@codis-178 ~]$ c="192.168.16.1-$a"
[xiaoda@codis-178 ~]$ echo "a=$a"
a=192.168.16.1-192.168.16.1
[xiaoda@codis-178 ~]$ echo "b=$b"
b=192.168.16.1-$a
[xiaoda@codis-178 ~]$ echo "c=${c}"
c=192.168.16.1-192.168.16.1-192.168.16.1

把命令的结果作为变量的内容进行赋值

[xiaoda@codis-178 ~]$ CMD=$(date + %F)
date: extra operand `%F'
Try `date --help' for more information.
[xiaoda@codis-178 ~]$ CMD=$(date +%F)
[xiaoda@codis-178 ~]$ echo $CMD
2017-07-04
[xiaoda@codis-178 ~]$ echo $(date +%F).tar.gz
2017-07-04.tar.gz
[xiaoda@codis-178 ~]$ echo $(CMD).tar.gz
-bash: CMD: command not found
.tar.gz
[xiaoda@codis-178 ~]$ echo $CMD.tar.gz
2017-07-04.tar.gz
[xiaoda@codis-178 ~]$ echo ${CMD}.tar.gz
2017-07-04.tar.gz

3.变量输出

单引号:内容原样输出,强引用
双引号:如果内容有命令、变量、转移符等,会解析出结果再输出,弱引用
反引号:引用命令

[xiaoda@codis-178 ~]$ echo 'today is date'
today is date
[xiaoda@codis-178 ~]$ echo 'today is `date`'
today is `date`
[xiaoda@codis-178 ~]$ echo "today is date"
today is date
[xiaoda@codis-178 ~]$ echo "today is `date`"
today is Tue Jul  4 15:34:38 CST 2017
[xiaoda@codis-178 ~]$ echo "today is $(date)"
today is Tue Jul  4 15:34:49 CST 2017

调用变量

[xiaoda@codis-178 ~]$ OLDBOY=testchars
[xiaoda@codis-178 ~]$ echo $OLDBOY
testchars
[xiaoda@codis-178 ~]$ echo '$OLDBOY'
$OLDBOY
[xiaoda@codis-178 ~]$ echo "$OLDBOY"
testchars

其他调用方式

[xiaoda@codis-178 ~]$ cat grep.log 
testchars
oldboy
[xiaoda@codis-178 ~]$ grep "$OLDBOY" grep.log 
testchars
[xiaoda@codis-178 ~]$ grep '$OLDBOY' grep.log 
[xiaoda@codis-178 ~]$ grep $OLDBOY grep.log 
testchars

[xiaoda@codis-178 ~]$ ETT=123

[xiaoda@codis-178 ~]$ awk 'BEGIN {print "$ETT"}'
$ETT
[xiaoda@codis-178 ~]$ awk 'BEGIN {print $ETT}'

[xiaoda@codis-178 ~]$ awk 'BEGIN {print '$ETT'}'
123
[xiaoda@codis-178 ~]$ awk 'BEGIN {print "'$ETT'"}'
123

posted @ 2017-08-30 17:36  BXBZ—边学边做  阅读(403)  评论(0编辑  收藏  举报