shell_script1,shellscript


1、简介 2、read 3、运算工具 4、if/then结构 5、while循环 6、for循环   一、简介 1、什么是shell shell是用户与系统交互作用的界面。shell是一种命令解释程序,同时也是一种高级程序设计语言 2、shell常见种类 Bourne Shell(/usr/bin/sh或/bin/sh) Bourne Again Shell(/bin/bash) C Shell(/usr/bin/csh) K Shell(/usr/bin/ksh) Shell for Root(/sbin/sh) 其中:Bash在日常工作中被广泛使用;同时,Bash也是大多数Linux系统默认的Shell; 3、shell局限性 1.1、需要耗费大量资源的任务,特别是对执行速度要求较高的场合 1.2、涉及大量的数学计算 1.3.、关键性应用(数据库,网站等) 1.4.、设计图形或者GUI的应用 1.5.、需要直接访问硬件 1.6.、开发闭源的应用(相对于开源) 4、基础 文件系统:Linux 的文件系统是一个包含了目录和文件的分层的组织结构,位于最顶端的叫做根目录(root directory),用斜杠/ 来表示 目录: 是一种包含目录项的文件,每个目录项中都包含了文件名 文件名: 目录的内容称为目录项,目录项包含了文件名,只有两种字符不允许出现在文件名中:斜杠,空字符(ASCII 值为0),斜杠用于分隔路径中的文件名,空字符用来表示路径的结尾。文件名的长度通常可以达到255个字符 路径: 一系列文件名连起来,用斜杠分隔,就叫做路径,路径用于表示文件的位置;以斜杠开头的路径叫做绝对路径,否则为相对路径,相对路径相对于当前工作目录 返回码:所有命令都有一个返回值,用0-255之间的整数来表示;脚本就是一个命令,它也有返回值,另外,脚本里面的函数也有返回值 退出脚本的正规方法是明确地用命令exit [code]来退出,如:exit 0;exit $?   二、read 1、定义 read是一个buildin命令,主要完成对参数的赋值,类似C语言中的scanf; 其不仅可以赋值变量,还可以赋值数组; 其输入不仅是屏幕,还可以是文件描述符 2、实操 # read a --输入字符串,将字符串赋值给变量 a;# echo $a hello world!! # read name sex age --同时分别给三个变量赋值,输入分隔符默认为 空格符 zhangsan male 50 # unset name sex age --删除变量 # IFS=';' --将 read 的输入分隔符改为 ';' # vim test.sh #!/bin/bash prompt="Please enter your name:" read -p "$prompt" name echo "Greetings $name"   三、运算工具 1、简单数学运算 # echo $((1+2**2-3*4/5%6)) --输出计算结果到屏幕 # a=$((1+2**2-3*4/5%6)) --把计算结果赋给变量a # ((a=1+2**2-3*4/5%6)) --同上 # echo "scale=3;10/3" | bc --bc计算器;保留小数点后三位 # echo "2^10" | bc --bc计算器;计算2的10次方 # awk 'BEGIN{print 1/3}' --awk计算 2、获取随机数 # od -N4 -tu4 /dev/urandom | sed -n '1s/.* //p' --获取一个7位数的号码 # od -Ad -w24 -tu4 /dev/urandom --获取随机数 # od -tu8 /dev/urandom --获取随机数 # od -tu8 /dev/urandom | sed -r 's/^[0-9]+\s+//' | sed -r 's/\s+/\n/g' |cut -b1-8 |sed -r -n '/^.{8}$/p' | sed 's/^/186/' |head -n5000 |vi - --获取一万个电话号码 # od -N40000 -tu4 /dev/urandom | sed -r 's/^[0-9]+(\s+)?//' | sed -r 's/\s+/\n/g' | grep -vE '^\s*$' > 10k_random --生成一万个随机数 # sed -r -n '/^.{8}$/p' 10k_random | head -n 100000 | sed 's/^/186/' > phone_num 3、获取数字序列 # seq 10 --获取1到10 # seq 1 2 10 --获取1到10,步长为2 # seq 10 100 --获取10到100 # seq 10 -1 1 --倒序,10到1   四、if/then实例 1、判断字符 "a" 是否等于 "A" # vim test.sh #!/bin/bash if [[ "a" = "A" ]]; then if [ "a" = "A" ]; then   echo "a equals A" else   echo "a no equals A" fi 2、判断/root/test/test.sh是否存在并具有可执行权限 # vim test.sh #!/bin/bash if test -x /root/test/test.sh;then   echo "/root/test/test.sh is executable" fi 3、判断系统是否存在 root 用户 # vim test.sh #!/bin/bash if grep -E --color=auto ^root: /etc/passwd; then   echo "user root exists" fi 4、判断文件类型 # vim test.sh #!/bin/bash function isSymbolicLink() {   file=$1   flag=$(ls -ld $file | cut -b1)   test "$flag" = "1"   return $? } file=$1 if isSymbolicLink $file; then   echo "Symbolic Link" elif test -d $file; then   echo "Directory file" elif test -f $file; then   echo "Regular file" elif test -b $file; then   echo "Block special" elif test -c $file; then   echo "Character special" elif test -p $file; then   echo "Named pipe" elif test -S $file; then   echo "Socket" else   echo "Unkown" fi 5、判断输入数字的大小 # vim test.sh #!/bin/bash num=$1 if test "$num" -lt 10 ;then   echo "小数字" elif test "$num" -lt 20;then   echo "中数字" elif test "$num" -lt 30;then   echo "大数字" else   echo "超大数字" fi 6、从标准输入获取一个数字,并根据其大小输出指定字符串 # vim test.sh #!/bin/bash echo -n "Enter a number: " read num if [ "$num" -lt 100 ]; then echo "小于100" elif [ "$num" -ge 100 -a "$num" -lt 200 ]; then echo "大于等于100小于200" elif [ "$num" -ge 200 -a "$num" -lt 300 ]; then echo "大于等于200小于300" else echo "其它数字" fi   五、wihle循环 1、认识 测试 while 关键字后面一条命令的返回码,条件为真时,程序读入while循环体中的指令;0为真。循环体如下: while [] --while 后面运行 [ ] 命令,测试 [ ] 命令的返回码 cat /filename | while read line       --while 后面运行read 命令,测试 read 命令的返回码 do ...... done 2、循环控制语句 continue --终止当前循环,开始下一个循环 break --终止当前循环,并退出循环体 exit --终止当前脚本 3、实例 # vim test.sh --区分 continue 与 break 的区别 #!/bin/bash while true do sleep 1 echo test continue/break echo done # vim test.sh --区分 break 与 exit 的区别 #!/bin/bash while test -e /data/test/test.sh do echo "exists" sleep 1 break/exit done echo "loop end" # vim test.sh --寻找我们给定的路径,直到寻到为止 #!/bin/bash if test $# -ne 1;then echo "wrong paraneter" >&2 exit fi path=$1 while test ! -e "$path" do sleep 1 echo "don't found!!!" done echo "found" # vim test.sh --脚本运行指定的时间,时间一到就退出;单位为 s #!/bin/bash if test $# -ne 1;then echo "Usage: $(basename $0) TIME" >&2 exit 1 fi timeLength=$1 beginTime=$(date +%s) endTime=$(( $beginTime + $timeLength )) --------------------------------------------------------- while test $(date +%s) -lt "$endTime" do echo "processing....." sleep 1 done --------------------------------------------------------- while true do if test $(date +%s) -ge "$endTime";then break fi echo "processing....." sleep 1 done --------------------------------------------------------- echo "time out" # vim test.sh --循环读取文件的每一行,并计算每行的字符个数 #!/bin/bash file=$1 --------------------------------------------------------- totalLines=$(wc -l < $file) line=1 while test $line -le $totalLines do lineData=$(sed -n "${line}p" $file) len=$(echo -n $lineData | wc -c) echo "line ${line}: $len" line=$(( $line + 1 )) done --------------------------------------------------------- line=1 while read lineData do len=$(echo $lineData | wc -c) echo "line ${line}: $len" line=$(( $line + 1 )) done < $file --------------------------------------------------------- # vim test.sh -- 打印出10行hello world;<(seq 10)相当于一个文件路径;称之为进程置换 #!/bin/bash while read num do echo "hello world" done < <(seq 10) --------------------------------------------------------- n=1 while [ $n -le 10 ] do echo "hello world" n=$((n+1)) done --------------------------------------------------------- # vim test.sh --创建一个不尽的循环,要求循环运行1分钟后自动终止 #!/bin/bash start_time=$(date +%s) while true do cur_time=$(date +%s) test $((cur_time - start_time)) -ge 10 && break time=$((cur_time - start_time)) echo "time is $time......" sleep 1 done # vim test.sh --先用while 循环创建100个.txt文件;再将所有文件后缀名改为 .html #!/bin/bash count=100 --------------------------------------------------------- seq $count | while read i do touch ${i}.txt done ls -1 *.txt | while read oldname do newname=$(echo $oldname | sed 's/.txt$/.html/') mv $oldname $newname done --------------------------------------------------------- while read i do touch ${i}.txt done < <( seq $count ) while read oldname do newname=$(echo $oldname | sed 's/.txt$/.html/') mv $oldname $newname done < <(ls -1 *.txt ) # vim test.sh --计算出1000 之内的偶数的和;不能使用管道,因为管道会生成子进程 #!/bin/bash sum=0 while read num do sum=$(( $sum + $num )) done < <(seq 2 2 998) echo "sum: $sum " # vim test.sh --批量添加100个邮件用户,用户名为u1 至u100,密码为abc,登录shell 为/sbin/nologin,只属于email组 #!/bin/bash password=abc group=email grep -Eq "^${group}:" /etc/group || groupadd $group while read i do useradd u${i} -N -g $group -s /sbin/nologin passwd u${i} --stdin <<< "$password" done < <(seq 100) # vim test.sh --批量删除刚刚创建的100个邮件用户 #!/bin/bash while read i do userdel -r u${i} done < <(seq 100   六、for 循环 在一系列由分隔符隔开的字符串中,按顺序每次取其中之一,当取完之后,循环结束 每次取出的字符串会被保存在一个变量中,可以在循环体内使用该变量 由分隔符隔开的字符串可以通过以下方法提供: 1、手动输入字符串,用于循环指定次数 # vim test.sh #!/bin/bash for i in 1 2 3 do echo $i don 2、由变量提供字符串 # vim test.sh #!/bin/bash FILES="/bin/bash /bin/ls /bin/cat" for file in $FILES do echo $file done 3、程序生成的字符串 # vim test.sh #!/bin/bash for n in $(seq 11) do   echo $n done 4、由shell 展开而生成,循环目录中的文件 # vim test.sh #!/bin/bash for f in /etc/init.d/* do echo $f done # vim test.sh 5、打印出/dev 下面所有以 loop 开头的文件的文件名 # vim test.sh #!/bin/bash for file in /dev/loop* do echo $(basename $file) done 6、计算出1000 之内的偶数的和 # vim test.sh #!/bin/bash sum=0 for number in $(seq 2 2 998) do   sum=$(( $sum + $number )) done echo "total: $sum" 7、批量添加100个邮件用户,用户名为u1 至u10,密码为abc,登录shell 为/sbin/nologin,只属于email组 # vim test.sh #!/bin/bash function checkGid() { gid=$1 if ! grep -qE "^$gid:" /etc/group; then groupadd $gid fi } function isRoot() { id=$(id | awk -F "[=(]" '{print $2}') if test "$id" -ne 0; then echo "must be root" >&2 exit 1 fi } count=10 namePrefix=u password=abc shell=/sbin/nologin gid=email isRoot checkGid $gid for num in $(seq $count) do name=${namePrefix}${num} useradd $name -s $shell -g $gid &> /dev/null if test $? -ne 0; then echo "failed to create $name" >&2 else echo "created successfully -- $name" echo "$password" | passwd --stdin $name &> /dev/null if test $? -ne 0; then echo "failed to change password for $name" >&2 else echo "password changed successfully -- $name" fi fi done 8、获取局域网内所有电脑的IP 和MAC 地址的对应表 # vim test.sh #!/bin/bash ipPrefix=192.168.1. startIp=1 endIp=254 for num in $(seq $startIp $endIp) do ip=${ipPrefix}$num ping -W1 -c1 $ip &>/dev/null & done wait arp -n | sed '/incomplete/d' | awk '{print $1,$3}'| cat | sort -n -t '.' -k4,4 | column -t 9、打印出九九乘法表 # vim test.sh #!/bin/bash for row in $(seq 9) do for col in $(seq $row) do echo -n "${col}x${row}=$(( $col * $row )) " done echo done | column -t 10、简陋计算器 #!/bin/bash varnum=3 if test "$#" -ne "$varnum"; then echo "wrong argument" >&2 exit 1 fi num1=$1 num2=$3 operator=$2 if test "$operator" = "x";then operator="*" fi if test "$operator" = "/" -a "$num2" = 0; then echo "division by 0" >& 2 exit 1 fi result=$(( $num1 $operator $num2 )) echo "$result" 注意:乘法的处理;被除数为0时的处理 11、把指定目录下的某些文件复制到指定的目录,要求: #1. 从命令行参数接收两个参数,分别是源目录和目标目录 #2. 如果源目录不存在,应该报错 #3. 如果目标目录不存在,就创建一个 #4. 如果目标已经存在但不是目录,就报错 #5. 只复制常规文件,软链接文件 #6. 对于常规文件,只复制大小小于1M 的文件 # vim /test.sh #!/bin/bash function displayHelp() { echo "Usage: $(basename $0) SRCDIR DSTDIR" } function getSize() { file=$1 size=$(ls -l $file | awk '{print $5}') echo $size } if test $# -ne 2; then displayHelp >&2 exit 1 fi srcdir=$1 dstdir=$2 if test ! -d "$srcdir"; then echo "$srcdir not exists or is not a directory" >&2 exit 1 fi if test ! -e "$dstdir"; then mkdir "$dstdir" fi if test ! -d "$dstdir"; then echo "$dstdir is not a directory" >&2 exit 1 fi for file in $srcdir/* do if test -L $file; then cp -a $file $dstdir/ elif test -f $file; then --------------------------------------------------- size=$(getSize $file) if test "$size" -lt 1048576; then cp -a $file $dstdir/ fi --------------------------------------------------- find $file -type f -size -1024k -exec cp -a {} $dstdir/ \; --------------------------------------------------- fi done

相关内容

    暂无相关文章