Linux&shell之结构化命令进阶

写在前面:案例、常用、归类、解释说明。(By Jim)

 for命令
重复一系列的命令是一种常见的编程实践。

#!/bin/bash
# basic for command
for test in A B C D E F G H I J K L M N O P Q
do
  echo The next letter is $test
done

结果:
The next letter is A
The next letter is B
The next letter is C
The next letter is D
The next letter is E
The next letter is F
The next letter is G
The next letter is H
The next letter is I
The next letter is J
The next letter is K
The next letter is L
The next letter is M
The next letter is N
The next letter is O
The next letter is P
The next letter is Q

每次for命令将列表中的下一个值赋值给变量test。变量$test可以像for命令语句中
使用其他脚本变量一样使用。最后一次迭代之后,变量$test在shell脚本的其他部分中仍然有效。它仍然是迭代的最后一个值。
如下:

#!/bin/bash
# basic for command
for test in A B C D
do
  echo The next letter is $test
done
echo "The last letter is $test"
test=Z
echo "Wait,now the letter is $test"

结果:
The next letter is A
The next letter is B
The next letter is C
The next letter is D
The last letter is D
Wait,now the letter is Z

单引号会引起问题,如下所示

#!/bin/bash
# basic for command
for test in I don't know if this'll work
do
  echo "word:$test"
done

结果:
word:I
word:dont know if thisll
word:work
由于单引号的存在出现问题了,有两种方法解决这个问题:
使用转义字符\来转义单引号;
或使用双引号来定义使用单引号的值。
两种方法都不是很好。

#!/bin/bash
# basic for command
for test in I don\'t know if this\'ll work
do
  echo "word:$test"
done
或
#!/bin/bash
# basic for command
for test in I "don't" know if "this'll" work  
do
  echo "word:$test"
done

如果有的数据之间有空格,可以通过双引号括起来
如下:

#!/bin/bash
# basic for command
for test in Nevada New Hampshire New Mexico New York North Carolina
do
  echo "Now going to $test"
done

结果:
Now going to Nevada
Now going to New
Now going to Hampshire
Now going to New
Now going to Mexico
Now going to New
Now going to York
Now going to North
Now going to Carolina
但是这并不是我们想要到结果,解决办法

#!/bin/bash
# basic for command
for test in Nevada "New Hampshire" "New Mexico" "New York" "North Carolina"
do
  echo "Now going to $test"
done

结果:
Now going to Nevada
Now going to New Hampshire
Now going to New Mexico
Now going to New York
Now going to North Carolina

从变量读取列表

#!/bin/bash
# using a variable to hold the list
list="Beijing Shanghai Guangzhou Shenzhen Nanjing Hangzhou"
list=$list" Suzhou"

for city in $list
do
  echo "Have you ever visited $city"
done

结果:
Have you ever visited Beijing
Have you ever visited Shanghai
Have you ever visited Guangzhou
Have you ever visited Shenzhen
Have you ever visited Nanjing
Have you ever visited Hangzhou
Have you ever visited Suzhou

读取命令中的值
生成列表中使用的值的另一种方法是使用命令的输出。
如下:

#!/bin/bash
# reading values from a file
file="cities"

for city in `cat $file`
do
  echo "Have you ever visited $city"
done

文件cities中的内容
Nanjing
Shang hai
Beijing
Shenzhen
Suzhou
Guangzhou
结果:
Have you ever visited Nanjing
Have you ever visited Shang
Have you ever visited hai
Have you ever visited Beijing
Have you ever visited Shenzhen
Have you ever visited Suzhou
Have you ever visited Guangzhou

问题,这里有空格的话,就会被分成两个输出比如Shang hai 被分成了两个Shang 和hai
怎么解决呢?下面讲解
默认情况下,bash shell将下面的字符看作字段分隔符:空格,制表符,换行符
可以在shell脚本中暂时更改环境变量IFS的值。例如:

#!/bin/bash
# reading values from a file
file="cities"

IFS=$'\n'
for city in `cat $file`
do
  echo "Have you ever visited $city"
done

这个时候就会把空格排除在外了

使用通配符读取目录

#!/bin/bash
#iterate through all the files in a directory

for file in /home/* /home/jiqing9006/shellscript/*
do
  if [ -d "$file" ]
  then
    echo "$file is a directory"
  elif [ -f "$file" ]
  then
    echo "$file is a file"
  fi
done

结果:
/home/jiqing9006 is a directory
/home/lost+found is a directory
/home/test is a directory
/home/wghan is a directory
/home/jiqing9006/shellscript/cities is a file
/home/jiqing9006/shellscript/test1 is a file
/home/jiqing9006/shellscript/test10 is a file
/home/jiqing9006/shellscript/test11 is a file
...

C式的for命令

#!/bin/bash
#testing the C-style for loop
for((i=1;i<=10;i++))
do
  echo "The next number is $i"
done

使用多个变量

#!/bin/bash
#testing the C-style for loop
for((a=1,b=10;a<=10;a++,b--))
do
  echo "$a-$b"
done

结果:
1-10
2-9
3-8
4-7
5-6
6-5
7-4
8-3
9-2
10-1

#!/bin/bash
#testing the C-style for loop
for((a=1,b=10;a<=10;a++,b--))
do
  echo $[ $a-$b ]
done

这样就会计算值了

While循环

#!/bin/bash
# while command test
var1=10
while [ $var1 -gt 0 ]
do
  echo $var1
  var1=$[ $var1 - 1 ]
done


多条件

#!/bin/bash
# while command test
var1=10
while echo $var1
      [ $var1 -gt 0 ]
do
  echo "This is inside the loop"
  var1=$[ $var1 - 1 ]
done

结果
10
This is inside the loop
9
This is inside the loop
8
This is inside the loop
7
This is inside the loop
6
This is inside the loop
5
This is inside the loop
4
This is inside the loop
3
This is inside the loop
2
This is inside the loop
1
This is inside the loop
0

until命令
until命令刚好与while命令相反。until命令需要制定一条测试命令,这条命令通常产生一个非零的退出状态。
只要测试命令的退出状态非零,bash就会执行列在循环当中的命令。一旦为0,循环停止。

#!/bin/bash
# using the until command
var1=100
until [ $var1 -gt 0 ]
do
  echo $var1
  var1=$[ $var1 -25 ]
done

将不会执行,因为符合了大于0的条件,这个与while是相反的

嵌套循环

#!/bin/bash
# nesting for loops
for((a = 1;a<=3;a++))
do
  echo "Starting loop $a:"
  for((b=1;b<=3;b++))
  do
    echo "Inside loop:$b"
  done
done

(这里面空格就没那么重要了,都识别的)

结果:
Starting loop 1:
Inside loop:1
Inside loop:2
Inside loop:3
Starting loop 2:
Inside loop:1
Inside loop:2
Inside loop:3
Starting loop 3:
Inside loop:1
Inside loop:2
Inside loop:3

混合循环

#!/bin/bash
# placing a for loop inside a while loop
var1=5
while [ $var1 -ge 0 ]
do
  echo "Outer loop:$var1"
  for((var2=1;var2<3;var2++))
  do
    var3=$[ $var1*$var2 ]
    echo "Inner loop:$var1*$var2 = $var3"
  done
  var1=$[$var1 -1]
done

结果:
Outer loop:5
Inner loop:5*1 = 5
Inner loop:5*2 = 10
Outer loop:4
Inner loop:4*1 = 4
Inner loop:4*2 = 8
Outer loop:3
Inner loop:3*1 = 3
Inner loop:3*2 = 6
Outer loop:2
Inner loop:2*1 = 2
Inner loop:2*2 = 4
Outer loop:1
Inner loop:1*1 = 1
Inner loop:1*2 = 2
Outer loop:0
Inner loop:0*1 = 0
Inner loop:0*2 = 0

再来看一下until与while结合的,准备晕吧少年

#!/bin/bash
#using until and while loops
var1=3
until [ $var1 -eq 0 ]
do
  echo "Outer loop:$var1"
  var2=1
  while [ $var2 -lt 5 ]
  do
    var3=`echo "scale=4;$var1/$var2"|bc`
    echo "  Inner loop:$var1/$var2 = $var3"
    var2=$[ $var2 +1 ]
  done
  var1=$[ $var1 - 1 ]
done

#外循环执行三次,内循环每执行四次
结果:
Outer loop:3
  Inner loop:3/1 = 3.0000
  Inner loop:3/2 = 1.5000
  Inner loop:3/3 = 1.0000
  Inner loop:3/4 = .7500
Outer loop:2
  Inner loop:2/1 = 2.0000
  Inner loop:2/2 = 1.0000
  Inner loop:2/3 = .6666
  Inner loop:2/4 = .5000
Outer loop:1
  Inner loop:1/1 = 1.0000
  Inner loop:1/2 = .5000
  Inner loop:1/3 = .3333
  Inner loop:1/4 = .2500
控制循环
break&continue

#!/bin/bash
# breaking out of a for loop

for var1 in 1 2 3 4 5 6 7 8 9 10
do
  if [ $var1 -eq 5 ]
  then
    break
  fi
  echo "number:$var1"
done
echo "The for loop is completed"

同样适用于while循环

#!/bin/bash
# breaking out of a while loop

var1=1

while [ $var1 -lt 10 ]
do
  if [ $var1 -eq 5 ]
  then
    break
  fi
  echo "number:$var1"
  var1=$[ $var1 + 1 ]
done
echo "The while loop is completed"

(循环到等于5的时候,就停止了)

跳出内循环

#!/bin/bash
# breaking out of an inner loop

for ((a=1;a<4;a++))
do
  echo "Outer loop:$a"
  for((b=1;b<100;b++))
  do
    if [ $b -eq 5 ]
    then
      break
    fi
    echo "  Inner loop:$b"
  done
done

结果:
Outer loop:1
  Inner loop:1
  Inner loop:2
  Inner loop:3
  Inner loop:4
Outer loop:2
  Inner loop:1
  Inner loop:2
  Inner loop:3
  Inner loop:4
Outer loop:3
  Inner loop:1
  Inner loop:2
  Inner loop:3
  Inner loop:4
  (跳出了内部循环,但是外部循环继续进行)

跳出外部循环
break n,默认n是1,代表跳出当前循环。如果将n设置为2,将停止外循环的下一级循环。

#!/bin/bash
# breaking out of an inner loop

for ((a=1;a<4;a++))
do
  echo "Outer loop:$a"
  for((b=1;b<100;b++))
  do
    if [ $b -eq 5 ]
    then
      break 2
    fi
    echo "  Inner loop:$b"
  done
done

结果:
Outer loop:1
  Inner loop:1
  Inner loop:2
  Inner loop:3
  Inner loop:4
  (跳出了外部循环,当然本次的循环也终止了)

continue命令

#!/bin/bash
# using the continue command

for((var1 = 1;var1<15;var1++))
do
  if [ $var1 -gt 5 ]&&[ $var1 -lt 10 ]
  then
    continue
  fi
  echo "Iteration number:$var1"
done

结果:
Iteration number:1
Iteration number:2
Iteration number:3
Iteration number:4
Iteration number:5
Iteration number:10
Iteration number:11
Iteration number:12
Iteration number:13
Iteration number:14
(循环没有终止,只是跳过了,继续下一次循环)


#!/bin/bash
# using the continue command

for((a=1;a<=5;a++))
do
  echo "Iteration $a:"
  for((b=1;b<=10;b++))
  do
    if [ $a -gt 3 ]&&[ $b -lt 5 ]
    then
      continue
    fi
    res=$[ $a*$b ]
    echo "The result of $a*$b is:$res"
  done
done

(这个时候,会在a为4时或5时,的大循环中断掉b小于5的几个计算)

#!/bin/bash
# using the continue command

for((a=1;a<=5;a++))
do
  echo "Iteration $a:"
  for((b=1;b<=10;b++))
  do
    if [ $a -gt 3 ]&&[ $b -lt 5 ]
    then
      continue 2
    fi
    res=$[ $a*$b ]
    echo "The result of $a*$b is:$res"
  done
done

(这里直接跳过a为4或5的内循环,连b大于5的内容也不显示了)

#!/bin/bash
# using the continue command

for((a=1;a<=5;a++))
do
  echo "Iteration $a:"
  for((b=1;b<=10;b++))
  do
    if [ $a -gt 3 ]&&[ $b -lt 5 ]
    then
      break
    fi
    res=$[ $a*$b ]
    echo "The result of $a*$b is:$res"
  done
done

(跳出a为4或5的内循环,效果等同于continue 2)

#!/bin/bash
# using the continue command

for((a=1;a<=5;a++))
do
  echo "Iteration $a:"
  for((b=1;b<=10;b++))
  do
    if [ $a -gt 3 ]&&[ $b -lt 5 ]
    then
      break 2
    fi
    res=$[ $a*$b ]
    echo "The result of $a*$b is:$res"
  done
done

(直接跳出之后的动作,跳出外循环也终止了)

处理循环的输出

#!/bin/bash
# using the continue command

for((a=1;a<=5;a++))
do
  echo "Iteration $a:"
  for((b=1;b<=10;b++))
  do
    if [ $a -gt 3 ]&&[ $b -lt 5 ]
    then
      break
    fi
    res=$[ $a*$b ]
    echo "The result of $a*$b is:$res"
  done
done > output.txt

(会将循环得到的结果写入output.txt文档下)

你可能感兴趣的:(linux)