一、函數(shù): 1.概述:Shell函數(shù)類似于Shell腳本,里面存放了一系列的指令,不過Shell的函數(shù)存在于內(nèi)存,而不是硬盤文件,所以速度很快,另外,Shell還能對(duì)函數(shù)進(jìn)行預(yù)處理,所以函數(shù)的啟動(dòng)比腳本更快。shell允許將一組命令集或語句形成一個(gè)可用塊,這些塊稱為shell函數(shù)。 2.語法: function 函數(shù)名() { 語句 [return] } 解析:所有函數(shù)在使用前必須定義。這意味著必須將函數(shù)放在腳本開始部分,直至shell解釋器首次發(fā)現(xiàn)它時(shí),才可以使用。調(diào)用函數(shù)僅使用其函數(shù)名即可。函數(shù)中的關(guān)鍵字“return”可以放到函數(shù)體的任意位置,通常用于返回某些值,Shell在執(zhí)行到return之后,就停止往下執(zhí)行,返回到主程序的調(diào)用行,return的返回值只能是0~256之間的一個(gè)整數(shù),返回值將保存到變量“$?”中。 3.shell函數(shù)的退出及刪除: 函數(shù)結(jié)束之后會(huì)返回調(diào)用函數(shù)的部分繼續(xù)執(zhí)行, 退出函數(shù)體:exit退出整個(gè)腳本、break語句來中斷函數(shù)的執(zhí)行。 shell中查詢函數(shù)及刪除: source或. 腳本 ##將函數(shù)載入到內(nèi)存 declare -f ##可以顯示定義的函數(shù)內(nèi)容 declare -F ##可以只顯示定義的函數(shù)名 unset -f ##可以從Shell內(nèi)存中刪除函數(shù) 4.變量: 5.案例: 案例一:函數(shù)的基本使用 [root@localhost ~]# vi func_linuxfan.sh #!/bin/bash read -p "輸入第一個(gè)數(shù):" ANum function linuxfan(){ echo "這個(gè)函數(shù)的功能是兩個(gè)數(shù)進(jìn)行相加:" read -p "輸入第二個(gè)數(shù):" BNum echo "您輸入的數(shù)是$ANum和$BNum! " SUM=$(expr $ANum + $BNum) echo "兩個(gè)數(shù)相加的結(jié)果是:$SUM" local A="哈哈!" echo "\$A是使用了local聲明的局部變量,在函數(shù)體內(nèi)情況是:$A" echo "\$ANum是在函數(shù)體外的變量,函數(shù)體內(nèi)調(diào)用它的情況是:$ANum" return 0 ##設(shè)置返回值 } linuxfan ##調(diào)用函數(shù) echo "\$A 在函數(shù)體外,結(jié)果是這樣的:$A" echo "\$BNum 在函數(shù)體外,結(jié)果是這樣的:$BNum" :wq [root@localhost ~]# chmod +x func_linuxfan.sh [root@localhost ~]# ./func_linuxfan.sh 輸入第一個(gè)數(shù):123 這個(gè)函數(shù)的功能是兩個(gè)數(shù)進(jìn)行相加: 輸入第二個(gè)數(shù):456 您輸入的數(shù)是123和456! 兩個(gè)數(shù)相加的結(jié)果是:579 $A是使用了local聲明的局部變量,在函數(shù)體內(nèi)情況是:哈哈 $ANum是在函數(shù)體外的變量,函數(shù)體內(nèi)調(diào)用它的情況是:123 $A 在函數(shù)體外,結(jié)果是這樣的: $B 在函數(shù)體外,結(jié)果是這樣的: [root@localhost ~]# source func_linuxfan.sh ##從腳本文件中載入函數(shù) [root@localhost bin]# declare -f ##顯示當(dāng)前shell中函數(shù)內(nèi)容 linuxfan () { 省略函數(shù)體內(nèi)的內(nèi)容。 } [root@localhost bin]# declare -F ##查看當(dāng)前shell中函數(shù)名稱 declare -f linuxfan [root@localhost bin]# unset -f linuxfan ##刪除當(dāng)前shell中的函數(shù) [root@localhost bin]# declare -F ##查看驗(yàn)證 案例二:函數(shù)參數(shù)的傳遞 函數(shù)可以通過位置變量傳遞參數(shù)。 函數(shù)名 參數(shù)1 參數(shù)2 參數(shù)3 參數(shù)4 ... [root@localhost ~]# vi fun-paramters.sh #!/bin/bash funparam(){ echo "\$1可以給函數(shù)傳遞第一個(gè)參數(shù),函數(shù)的第一個(gè)參數(shù):$1" echo "\$2可以給函數(shù)傳遞第二個(gè)參數(shù),函數(shù)的第二個(gè)參數(shù):$2" echo "\$7可以給函數(shù)傳遞第七個(gè)參數(shù),函數(shù)的第七個(gè)參數(shù):$7" echo "\${10}可以給函數(shù)傳遞第十個(gè)參數(shù),函數(shù)的第十個(gè)參數(shù):${10}" echo "\${11}可以給函數(shù)傳遞第十一個(gè)參數(shù),函數(shù)的第十一個(gè)參數(shù):${11}" echo "函數(shù)有$#個(gè)參數(shù),函數(shù)參數(shù)具體內(nèi)容是$*" } funparam 1 2 3 4 5 6 11 8 9 66 99 echo "腳本后的參數(shù)\$1是$1;" echo "腳本后的參數(shù)\$2是$2;" echo "腳本后的參數(shù)\$3是$3;" echo "腳本后的參數(shù)\$4是$4;" echo "腳本后的參數(shù)\$5是$5;" :wq [root@localhost ~]# chmod +x fun-paramters.sh [root@localhost ~]# ./fun-paramters.sh a b c d f e $1可以給函數(shù)傳遞第一個(gè)參數(shù),函數(shù)的第一個(gè)參數(shù):1 $2可以給函數(shù)傳遞第一個(gè)參數(shù),函數(shù)的第一個(gè)參數(shù):2 $7可以給函數(shù)傳遞第一個(gè)參數(shù),函數(shù)的第一個(gè)參數(shù):11 ${10}可以給函數(shù)傳遞第一個(gè)參數(shù),函數(shù)的第一個(gè)參數(shù):66 ${11}可以給函數(shù)傳遞第一個(gè)參數(shù),函數(shù)的第一個(gè)參數(shù):99 函數(shù)有11個(gè)參數(shù),函數(shù)參數(shù)具體內(nèi)容是1 2 3 4 5 6 11 8 9 66 99 腳本后的參數(shù)\$1是a; 腳本后的參數(shù)\$1是b; 腳本后的參數(shù)\$1是c; 腳本后的參數(shù)\$1是d; 腳本后的參數(shù)\$1是f; [root@localhost ~]# cat test.sh ##將shell命令行中的參數(shù)傳遞給函數(shù) #!/bin/bash function test(){ echo "$1 $2 $3" } test $* [root@localhost ~]# ./ test.sh a b c a b c 案例三: 擴(kuò)展學(xué)習(xí)(更多練習(xí)): 向系統(tǒng)學(xué)習(xí)函數(shù)的使用: http://www.cnblogs.com/image-eye/archive/2011/10/26/2220405.html ##請(qǐng)大家閱讀/etc/init.d/funcations詳解,至少搞懂daemon和killproc兩個(gè)函數(shù)的作用。然后靜下心來閱讀這個(gè)腳本,并給它添加注釋你能學(xué)會(huì)很多東西: [root@localhost ~]# cat /etc/init.d/vsftpd ##這是vsftpd的啟動(dòng)腳本,非常經(jīng)典 #!/bin/bash # ### BEGIN INIT INFO # Provides: vsftpd # Required-Start: $local_fs $network $named $remote_fs $syslog # Required-Stop: $local_fs $network $named $remote_fs $syslog # Short-Description: Very Secure Ftp Daemon # Description: vsftpd is a Very Secure FTP daemon. It was written completely from # scratch ### END INIT INFO # vsftpd This shell script takes care of starting and stopping # standalone vsftpd. # # chkconfig: - 60 50 # description: Vsftpd is a ftp daemon, which is the program \ # that answers incoming ftp service requests. # processname: vsftpd # config: /etc/vsftpd/vsftpd.conf # Source function library. . /etc/rc.d/init.d/functions # Source networking configuration. . /etc/sysconfig/network RETVAL=0 prog="vsftpd" start() { # Start daemons. # Check that networking is up. [ ${NETWORKING} = "no" ] && exit 1 [ -x /usr/sbin/vsftpd ] || exit 1 if [ -d /etc/vsftpd ] ; then CONFS=`ls /etc/vsftpd/*.conf 2>/dev/null` [ -z "$CONFS" ] && exit 6 PROC_FAILED=0 for i in $CONFS; do site=`basename $i .conf` echo -n $"Starting $prog for $site: " daemon /usr/sbin/vsftpd $i RETVAL=$? echo if [ $RETVAL -eq 0 ] && [ ! -f /var/lock/subsys/$prog ]; then touch /var/lock/subsys/$prog elif [ $RETVAL -ne 0 ]; then ps -FC vsftpd | grep "$i" > /dev/null RETVAL=$? if [ $PROC_FAILED -eq 0 ] && [ $RETVAL -ne 0 ]; then PROC_FAILED=1 fi fi done if [ $RETVAL -eq 0 ] && [ $PROC_FAILED -ne 0 ]; then RETVAL=1 fi else RETVAL=1 fi return $RETVAL } stop() { # Stop daemons. echo -n $"Shutting down $prog: " killproc $prog RETVAL=$? echo [ $RETVAL -eq 0 ] && rm -f /var/lock/subsys/$prog return $RETVAL } # See how we were called. case "$1" in start) start ;; stop) stop ;; restart|reload) stop start RETVAL=$? ;; condrestart|try-restart|force-reload) if [ -f /var/lock/subsys/$prog ]; then stop start RETVAL=$? fi ;; status) status $prog RETVAL=$? ;; *) echo $"Usage: $0 {start|stop|restart|try-restart|force-reload|status}" exit 1 esac exit $RETVAL 案例四:測(cè)試主機(jī)訪問url路徑 二、shell數(shù)組: 1.概述:數(shù)組就是一組數(shù)據(jù)類型相同集合; 2.數(shù)組的定義及使用: [root@localhost bin]# arr1=() ##定義空元素?cái)?shù)組 [root@localhost bin]# arr2=(1 2 3 4 5 6) ##定義數(shù)字元素的數(shù)組 [root@localhost bin]# arr3=("有夢(mèng)想" "向往遠(yuǎn)方和詩" "高薪就業(yè)" "靜心沉迷于學(xué)習(xí)!") ##定義字符串元素的數(shù)組,注意用引號(hào)(單、雙引號(hào)皆可) [root@localhost bin]# echo ${arr2[0]} 1 [root@localhost bin]# echo ${arr2[3]} 4 [root@localhost bin]# echo ${arr3[3]} 靜心沉迷于學(xué)習(xí)! [root@localhost bin]# echo ${arr3[*]} 有夢(mèng)想 向往遠(yuǎn)方和詩 高薪就業(yè) 靜心沉迷于學(xué)習(xí)! [root@localhost bin]# for i in ${arr3[*]};do echo $i;done 有夢(mèng)想 向往遠(yuǎn)方和詩 高薪就業(yè) 靜心沉迷于學(xué)習(xí)! [root@localhost bin]# echo ${#arr3[*]} ##獲取數(shù)組的長(zhǎng)度 4 [root@localhost bin]# echo ${#arr3[@]} 4 數(shù)組賦值格式:數(shù)組名[下標(biāo)]=值,如果下標(biāo)不存在,則新增數(shù)組元素; 下標(biāo)已有,則覆蓋值。 [root@localhost bin]# echo ${arr2[*]} ##獲取數(shù)組所有元素 1 2 3 4 5 6 [root@localhost bin]# arr2[6]=8 ##下標(biāo)為6(第七個(gè))的內(nèi)容為8,添加元素 [root@localhost bin]# echo ${arr2[*]} ##驗(yàn)證 1 2 3 4 5 6 8 [root@localhost bin]# arr2[2]=8 ##修改下標(biāo)為2的元素為8,覆蓋原有值 [root@localhost bin]# echo ${arr2[*]} ##驗(yàn)證 1 2 8 4 5 6 8 數(shù)組分片的格式:${數(shù)組名[*或@]:起始位:長(zhǎng)度},截取部分?jǐn)?shù)組,返回字符串,中間用空格分隔;將結(jié)果使用“()”,則得到新的切片數(shù)組。 [root@localhost bin]# echo ${#arr3[*]} 4 [root@localhost bin]# echo ${arr3[*]:0:1} 有夢(mèng)想 [root@localhost bin]# echo ${arr3[*]:2:2} 高薪就業(yè) 靜心沉迷于學(xué)習(xí)! [root@localhost bin]# arr4=(${arr3[*]:2:2}) [root@localhost bin]# echo ${arr4[*]} 高薪就業(yè) 靜心沉迷于學(xué)習(xí)! 數(shù)組替換元素的格式:${數(shù)組名[*或@]/查找字符/替換字符}, 不會(huì)修改原數(shù)組;如需修改的數(shù)組,將結(jié)果使用“()”賦給新數(shù)組。 [root@localhost bin]# echo ${arr2[*]} 1 2 8 4 5 6 8 [root@localhost bin]# echo ${arr2[*]/4/9} 1 2 8 9 5 6 8 [root@localhost bin]# arr5=${arr2[*]/4/9} [root@localhost bin]# echo ${arr5[*]} 1 2 8 9 5 6 8 刪除數(shù)組的格式:unset 數(shù)組,清除整個(gè)數(shù)組; unset 數(shù)組[下標(biāo)],清除單個(gè)元素。 [root@localhost bin]# echo ${arr3[*]} 有夢(mèng)想 向往遠(yuǎn)方和詩 高薪就業(yè) 靜心沉迷于學(xué)習(xí)! [root@localhost bin]# unset arr3[0] [root@localhost bin]# echo ${arr3[*]} 向往遠(yuǎn)方和詩 高薪就業(yè) 靜心沉迷于學(xué)習(xí)! [root@localhost bin]# unset arr3 [root@localhost bin]# echo ${arr3[*]} 三、shell當(dāng)中括號(hào)的使用: 1.單小括號(hào)(): 命令組:組合多條命令一起執(zhí)行,并按照順序執(zhí)行。 [root@www ~]# (umask 0077;mkdir -p test;ls -ld test) 替換命令:效果等于反撇,在命令中執(zhí)行命令,并將執(zhí)行結(jié)果交給命令處理。 [root@www ~]# rpm -qf $(which convert) ##查詢命令的安裝包 ImageMagick-6.5.4.7-6.el6_2.x86_64 用于初始化數(shù)組:如array=(a b c d) [root@www ~]# array=(a b c d) 2.雙小括號(hào)(()) 計(jì)算其他進(jìn)制(二、八、十六)的數(shù)到十進(jìn)制: [root@www ~]# echo $((2#11)) ##二轉(zhuǎn)十 3 [root@www ~]# echo $((8#11)) ##八轉(zhuǎn)十 9 [root@www ~]# echo $((16#11)) ##十六轉(zhuǎn)十 17 [root@www ~]# echo $((16#1f)) 31 [root@localhost bin]# echo $((3+2)) ##加減乘除取摸運(yùn)算 5 [root@localhost bin]# echo $((3-2)) 1 [root@localhost bin]# echo $((3*2)) 6 [root@localhost bin]# echo $((3/2)) 1 [root@localhost bin]# echo $((3%2)) 1 重新定義變量: [root@www ~]# a=5;((a++));echo $a 6 算術(shù)運(yùn)算比較,雙括號(hào)內(nèi)的變量可以不使用$,表達(dá)式用分號(hào)分開: [root@www ~]# for i in {0..4};do echo $i;done [root@www ~]# for i in $(seq 0 4);do echo $i;done [root@www ~]# for ((i=0;i<5;i++));do echo $i;done ##上述三種都是一樣的效果 [root@www ~]# i=10 [root@www ~]# if ((i>5));then echo $i;fi [root@www ~]# if [ $i -gt 5 ];then echo $i;fi ##兩個(gè)if的效果相同 3.中括號(hào)[ ] 條件表達(dá)式 [root@www ~]# [ -f /etc/hosts ]&&echo ok [root@www ~]# test -f /etc/hosts &&echo ok 字符范圍。用作正則表達(dá)式的一部分,描述一個(gè)匹配的字符范圍。 [root@www ~]# i=1 ##case中使用的[0-9]|[a-z]|[A-Z]表示正則 [root@www ~]# case $i in [0-9]) echo "number"; ;; [a-z]|[A-Z]) echo "alph"; ;; esac 4.雙中括號(hào)[[ ]] ①[[是 bash 程序語言的關(guān)鍵字。并不是一個(gè)命令,[[ ]] 結(jié)構(gòu)比[ ]結(jié)構(gòu)更加通用。在[[和]]之間所有的字符都不會(huì)發(fā)生文件名擴(kuò)展或者單詞分割,但是會(huì)發(fā)生參數(shù)擴(kuò)展和命令替換。 ②支持字符串的模式匹配,使用=~操作符時(shí)甚至支持shell的正則表達(dá)式。字符串比較時(shí)可以把右邊的作為一個(gè)模式,而不僅僅是一個(gè)字符串,比如[[ hello == hell? ]],結(jié)果為真。[[ ]] 中匹配字符串或通配符,不需要引號(hào)。 ③使用[[ ... ]]條件判斷結(jié)構(gòu),而不是[ ... ],能夠防止腳本中的許多邏輯錯(cuò)誤。比如,&&、||、<和> 操作符能夠正常存在于[[ ]]條件判斷結(jié)構(gòu)中,但是如果出現(xiàn)在[ ]結(jié)構(gòu)中的話,會(huì)報(bào)錯(cuò)。比如可以直接使用if [[ $a != 1 && $a != 2 ]], 如果不使用雙括號(hào), 則為if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ]。 [root@www ~]# a=3 [root@www ~]# if [ $a != 1 && $a != 2 ];then echo $a;fi ##報(bào)錯(cuò) -bash: [: missing `]' [root@www ~]# if [[ $a != 1 && $a != 2 ]];then echo $a;fi ##成功執(zhí)行 3 [root@www ~]# if [ $a != 1 ] && [ $a != 2 ];then echo $a;fi ##執(zhí)行成功 3 5.花括號(hào){ } 常規(guī)用法: [root@www ~]# touch {a..z}.mp{3..5} ##..表示分割順序文件列表 [root@www ~]# ls {{a..e},h,y,z}.mp4 a.mp4 b.mp4 c.mp4 d.mp4 e.mp4 h.mp4 y.mp4 z.mp4 定義函數(shù):代碼塊,又被稱為內(nèi)部組,這個(gè)結(jié)構(gòu)事實(shí)上創(chuàng)建了一個(gè)匿名函數(shù) 。與小括號(hào)中的命令不同,花括號(hào)內(nèi)的命令不會(huì)新開一個(gè)子shell運(yùn)行,即腳本余下部分仍可使用括號(hào)內(nèi)變量。 字符串提取和替換: ${var:num},${var:num1:num2},${var/pattern/pattern},${var//pattern/pattern} 第一種模式:${var:num},這種模式時(shí),shell在var中提取第num個(gè)字符到末尾的所有字符。若num為正數(shù),從左邊0處開始;若num為負(fù)數(shù),從右邊開始提取字串,但必須使用在冒號(hào)后面加空格或一個(gè)數(shù)字或整個(gè)num加上括號(hào),如${var: -2}、${var:1-3}或${+var:(-2)}。 [root@www ~]# var=www.linuxfan.cn [root@www ~]# echo ${var:4} linuxfan.cn [root@www ~]# echo ${var:(-2)} cn 第二種模式:${var:num1:num2},num1是位置,num2是長(zhǎng)度。表示從$var字符串的第$num1個(gè)位置開始提取長(zhǎng)度為$num2的子串。不能為負(fù)數(shù)。 [root@www ~]# var=www.linuxfan.cn [root@www ~]# echo ${var:4:5} linux [root@www ~]# echo ${var:1:3} ww. [root@www ~]# echo ${var:0:3} www 第三種模式:${var/pattern/pattern}表示將var字符串的第一個(gè)匹配的pattern替換為另一個(gè)pattern。。 [root@www ~]# var=www.linuxfan.cn [root@www ~]# echo ${var/www/dns} dns.linuxfan.cn 第四種模式:${var//pattern/pattern}表示將var字符串中的所有能匹配的pattern替換為另一個(gè)pattern。 [root@www ~]# var=www.linuxfan.cn [root@www ~]# echo ${var/n/N} www.liNuxfan.cn [root@www ~]# echo ${var//n/N} www.liNuxfaN.cN 6.多條命令執(zhí)行 單小括號(hào):(cmd1;cmd2;cmd3)新開一個(gè)子shell順序執(zhí)行命令cmd1,cmd2,cmd3, 各命令之間用分號(hào)隔開, 最后一個(gè)命令后可以沒有分號(hào)。 [root@www tmp]# (touch index.html;rm -rf index.html;ls -l;) 單大括號(hào):{ cmd1;cmd2;cmd3;} 在當(dāng)前shell順序執(zhí)行命令cmd1,cmd2,cmd3, 各命令之間用分號(hào)隔開, 最后一個(gè)命令后必須有分號(hào), 括號(hào)兩側(cè)必須有空格。 [root@www tmp]# {touch index.html;rm -rf index.html;ls -l;} ##開始的{后無空格報(bào)錯(cuò) -bash: syntax error near unexpected token `}' [root@www tmp]# { touch index.html;rm -rf index.html;ls -l } [root@www tmp]# { touch index.html;rm -rf index.html;ls -l; } ##最后一條命令必須; 總用量 0 注:對(duì){}和()而言, 括號(hào)中的重定向符只影響該條命令,而括號(hào)外的重定向符影響到括號(hào)中的所有命令。 |
|