shell编程之从环境到常用命令
1.shell介绍
定义、在线工具
1.百科定义
2.通俗介绍
面向过程:Shell, C dev ops
面向对象: JAVA, Python, perl, C++
变量:内存空间,命名
内存:编址的存储单元
shell: 外壳
GUI:Gnome, KDE, Xfce
CLI: sh, csh, ksh, bash, tcsh, zsh
3.登录式shell如何读取配置文件?
/etc/profile --> /etc/profile.d/*.sh --> ~/.bash_profile --> ~/.bashrc --> /etc/bashrck
Source /etc/bashrc
非登录式shell如何配置文件?
~/.bashrc --> /etc/basrc --> /etc/profile.d/*.sh
2.shell变量
shell变量
1.定义变量时,变量名不加美元符号($,PHP语言中变量需要),如:
name="hhh"
2.定义规则:变量名和等号之间不能有空格。
同时,变量名的命名须遵循如下规则:
1)命名只能使用英文字母,数字和下划线,首个字符不能以数字开头;
2)中间不能有空格,可以使用下划线(_);
3)不能使用标点符号;
4)不能使用bash里的关键字(可用help命令查看保留关键字)。
3.**变量类型:*事先确定数据的存储格式和长度
1)面向对象 字符 数值 整型 浮点型 详见shell字符串
2) 运行shell时,会同时存在三种变量:
1 局部变量(本地变量) 局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。
2 *环境变量 所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。
3 *shell变量(位置变量、特殊变量)shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行。
变量名称:
1、只能包含字母、数字和下划线,并且不能数字开头;
2、不应该跟系统中已有的环境变量重名;
3、最好做到见名知义;
本地变量:
set VARNAME=VALUE: 作用域为整个bash进程;
局部变量:
local VARNAME=VALUE:作用域为当前代码段;
环境变量:作用域为当前shell进程及其子进程;
export VARNAME=VALUE
VARNAME=VALUE
export VARNAME
“导出”
位置变量:
$1, $2, …
Shift
特殊变量:
$?: 上一个命令的执行状态返回值;
程序执行,可能有两类返回值:
程序执行结果
程序状态返回代码(0-255)
0: 正确执行
1-255:错误执行,1,2,127系统预留;
$#
$*
$@
$$
$0
撤消变量:
unset VARNAME
查看当shell中变量:
Set (可以省略)
查看当前shell中的环境变量:
printenv
env
export
4.使用变量
1)使用定义过的变量,在变量名前加$即可
name="hhh"
echo $name
echo ${name} #推荐加花括号以便于解释器识别变量的边界
2)只读变量
使用 readonly 命令可以将变量定义为只读变量,只读变量的值不能被改变。
#!/bin/bash
myUrl="http://www.google.com"
readonly myUrl
myUrl="http://www.smartisan.com"
运行,尝试更改只读变量,结果报错:
/bin/sh: NAME: This variable is read only.
3)删除变量
使用 unset 命令可以删除变量。变量被删除后不能再次使用。unset 命令不能删除只读变量。语法:
unset variable_name
5.命令替换: $(COMMAND)
, 反引号:COMMAND
把命令中某个子命令替换为其执行结果的过程
file-2013-02-28-14-53-31.txt
bash支持的引号:
``: 命令替换
“”: 强引用,可以实现变量替换
‘’: 弱引用,不完成变量替换
文件名通配, globbing
*: 任意长度的任意字符
?:任意单个字符
[ ]:匹配指定范围内的任意单个字符
[abc], [a-m], [a-z], [A-Z], [0-9], [a-zA-Z], [0-9a-zA-Z]
[:space:]:空白字符
[:punct:]:标点符号
[:lower:]:小写字母
[:upper:]: 大写字母
[:alpha:]: 大小写字母
[:digit:]: 数字
[:alnum:]: 数字和大小写字母
[[:alpha:]]*[[:space:]]*[^[:alpha:]]
shell字符串
字符串是shell编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号。单双引号的区别跟PHP类似。
1.单引号
str='this is a string'
单引号字符串的限制:
单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
2.双引号
your_name='hhh'
str="Hello, I know you are \"$your_name\"! \n"
echo -e $str
输出结果为:
Hello, I know you are "hhh"!
双引号的优点:
双引号里可以有变量;双引号里可以出现转义字符。
3.拼接字符串
your_name="hhh"
使用双引号拼接
greeting="hello, "$your_name" !"
greeting_1="hello, ${your_name} !"
echo $greeting $greeting_1
使用单引号拼接
greeting_2='hello, '$your_name' !'
greeting_3='hello, ${your_name} !'
echo $greeting_2 $greeting_3
输出结果为:
hello, hhh ! hello, hhh !
hello, hhh ! hello, ${your_name} !
4.获取字符串长度
string="abcd"
echo ${#string} #输出 4
提取子字符串
以下实例从字符串第 2 个字符开始截取 4 个字符:
string="hthzkjh is a great site"
echo ${string:1:4} # 输出 thzk
查找子字符串
查找字符 i 或 o 的位置(哪个字母先出现就计算哪个):
注意: 以上脚本中 ` 是反引号,而不是单引号 '
5.字符串截取
#、## 表示从左边开始删除;
一个 # 表示从左边删除到第一个指定的字符;
两个 # 表示从左边删除到最后一个指定的字符;
%、%% 表示从右边开始删除;
一个 % 表示从右边删除到第一个指定的字符;
两个 % 表示从左边删除到最后一个指定的字符。
删除包括了指定的字符本身。
实例:
#!/bin/bash
# 字符串截取(界定字符本身也会被删除)
str="https://blog.csdn.net/qq_40993864"
echo "str : ${str}"
echo "str#*/ : ${str#*/}" # 从 字符串开头 删除到 左数第一个'/'
echo "str##*/ : ${str##*/}" # 从 字符串开头 删除到 左数最后一个'/'
echo "str%/* : ${str%/*}" # 从 字符串末尾 删除到 右数第一个'/'
echo "str%%/* : ${str%%/*}" # 从 字符串末尾 删除到 右数最后一个'/'
echo
echo "str#/* : ${str#/*}" # 无效果
echo "str##/* : ${str##/*}" # 无效果
echo "str%*/ : ${str%*/}" # 无效果
echo "str%%*/ : ${str%%*/}" # 无效果
输出结果:
str : https://blog.csdn.net/qq_40993864
str#*/ : /blog.csdn.net/qq_40993864
str##*/ : qq_40993864
str%/* : https://blog.csdn.net
str%%/* : https:
str#/* : https://blog.csdn.net/qq_40993864
str##/* : https://blog.csdn.net/qq_40993864
str%*/ : https://blog.csdn.net/qq_40993864
str%%*/ : https://blog.csdn.net/qq_40993864
shell数组
1.环境:bash支持一维数组(不支持多维数组),并且没有限定数组的大小。类似于 C 语言,数组元素的下标由 0 开始编号。获取数组中的元素要利用下标,下标可以是整数或算术表达式,其值应大于或等于 0。
2.定义:在 Shell 中,用括号来表示数组,数组元素用"空格"符号分割开。
3.数组
1)读取数组
${数组名[下标]}
如:valuen=${array_name[n]}
使用 @ 符号可以获取数组中的所有元素,例如:
echo ${array_name[@]}
2)获取数组长度
获取数组长度的方法与获取字符串长度的方法相同
# 取得数组元素的个数
length=${#array_name[@]}
# 或者
length=${#array_name[*]}
# 取得数组单个元素的长度
lengthn=${#array_name[n]}
变量:内存空间 字符型
数组:连续的内存空间
元素:存储在数组中的每一个变量称为元素,利用索引来标示
${AA[0]}
声明数组:declare -a AA
赋值方法(1):
AA[0]=jerry
AA[1]=tom
AA[2]=wendy
AA[6]=natasha
赋值方法(2):
AA=(jerry tom wendy)
AA=([0]=jerry [1]=tom [2]=wendy [6]=natasha)
AA[3]=nikita
Echo ${AA[0]}
随机产生数组元素
#!/bin/bash
#
AA=([0]=jerry [1]=tom [2]=wendy [6]=natasha)
For I in{1..20};do
INDEX=$[$RANDOM%7]
Echo $[AA[$INDEX]]
Done
shell数组实例
找出一组数据中的最大数
#!/bin/bash
#
ARRAY=(34 73 25 91 40 22 67 98 99 53)
Declare -i MAX=${ARRAY[0]}
For I in `seq 1 $[${#ARRY[*]}-1]`;do
If [ $MAX -lt ${ARRAY[$I]}];then
MAX=${ARRAY[$I]}
Fi
Done
Echo $MAX
随机生成数组:
#!/bin/bash
For J in {0..9};do
ARRAY[$J]=$RANDOM
Echo -n “$ARRAY[$J] ”
sleep
Done
Echo
For I in `seq 1 $[${#ARRY[*]}-1]`;do
If [ $MAX -lt ${ARRAY[$I]}];then
MAX=${ARRAY[$I]}
Fi
Done
Echo The max is $MAX
写一个脚本:
随机从同学中选择一位同学回答问题
生成一个数组:
1、数组的元素个数为1-39
read
2、数组元素不能相同
3、显示此数组各元素的值
#!/bin/bash
read -p "The element numbers[1-39]:" ELENUM
function COMELE {
for J in `seq 0 $[${#ARRAY[*]}-1]`;do
if [ $1 -eq ${ARRAY[$J]} ];then
return 1
fi
done
return 0
}
for I in `seq 0 $[$ELENUM-1]`;do
while true;do
ELEMENT=$[$RANDOM%40]
COMELE $ELEMENT
if [ $? -eq 0 ];then
break
fi
done
ARRAY[$I]=$ELEMENT
echo "${ARRAY[$I]}"
done
比较专业的生产检查URL地址的脚本(shell数组方法):
[root@oldboy ~]# cat check_url.sh
#!/bin/bash
this script is created by oldboy.
e_mail:31333741@qq.com
qqinfo:49000448
function:check web url
version:1.1
. /etc/init.d/functions
url_list=(
http://etiantian.org
http://www.linuxpeixun.com
http://oldboy.blog.51cto.com
)
function wait()
{
echo -n '3秒后,执行该操作.';
for ((i=0;i<3;i++))
do
echo -n ".";sleep 1
done
echo
}
function check_url()
{
wait
echo 'check url...'
for ((i=0; i<${#url_list[*]}; i++))
do
HTTP/1.1 200 OK
judge=($(curl -I -s ${url_list[$i]}|head -1|tr "\r" "\n"))
if [[ "${judge[1]}" == '200' && "${judge[2]}"=='OK' ]]
then
action "${url_list[$i]}" /bin/true
else
action "${url_list[$i]}" /bin/false
fi
done
}
check_url
shell注释
以 # 开头的行就是注释,会被解释器忽略。
通过每一行加一个 # 号设置多行注释,像这样:
#--------------------------------------------
##### 用户配置区 开始 #####
#
#
# 这里可以添加脚本描述信息
#
#
##### 用户配置区 结束 #####
如果在开发过程中,遇到大段的代码需要临时注释起来,过一会儿又取消注释,怎么办呢?
每一行加个#符号太费力了,可以把这一段要注释的代码用一对花括号括起来,定义成一个函数,没有地方调用这个函数,这块代码就不会执行,达到了和注释一样的效果。
多行注释
多行注释还可以使用以下格式:
:<<EOF
注释内容...
注释内容...
注释内容...
EOF
EOF 也可以使用其他符号:
:<<'
注释内容...
注释内容...
注释内容...
'
:<<!
注释内容...
注释内容...
注释内容...
!
3.shell传递参数
4.shell运算符
加减乘除
# !/bin/bash
a=88 b=22
val=`expr $a + $b` # 加法,注意运算符与变量之间要有空格
echo "加 $val"
val=`expr $a - $b`
echo "减 $val"
val=`expr $a \* $b` #乘法是\*,注意之前要加一个反斜杠\
echo "乘法 $val"
val=`expr $a / $b` #除法是/
echo 除法 $val"
val=`expr $b % $a` # 取余是#
echo "取余 $val"
val=`expr b = $a` #赋值 =
echo "赋值 $val"
if [ $a == $b ] # if...then...fi是shell的循环语句
then
echo "a等于b"
fi
if [ $a != $b ]
then
echo "a不等于b
"
fi
5.命令
echo命令
printf命令
test命令
6.shell流程控制
7.shell函数
8.shell输入/输出、重定向
9.shell文件包含
10.参考来源
参考来源1
参考来源2:自己的查找及老师教授。