Shell脚本系列(五)

目录
  1. Shell脚本系列(一)
  2. Shell脚本系列(二)
  3. Shell脚本系列(三)
  4. Shell脚本系列(四)
  5. Shell脚本系列(五)
  6. Shell脚本系列(六)
  7. Shell脚本系列(七)
  8. Shell脚本系列(八)

1、函数

Shell 可以用户定义函数,然后在Shell脚本中可以随便调用。

格式 :

[ function ] funname [()]
{
    action;
    [return int;]
}

说明:

  1)可以带function fun() 定义,也可以直接fun() 定义,不带任何参数;

  2)参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。 return后跟数值n(0-255)——特别注意它的返回值范围

  3)函数的定义必须放在函数调用的前面。这意味着必须将函数放在脚本开始部分,直至Shell解释器首次发现它时,才可以使用。调用函数仅使用其函数名即可;

  4)可以使用 “$?” 来获取函数的返回结果。但是需要注意:$? 仅对其上一条指令负责,一旦函数返回后其返回值没有立即保存入参数,那么其返回值将不再能通过 $? 获得

eg:

# 无参无返回
demoFun(){
    echo "这是我的第一个Shell函数";
}

echo "函数开始执行";
demoFun;
echo "函数执行完毕";


# 无参有返回
funWithReturn(){
    echo "这个函数的作用是计算两个数组的和";
    echo "请输入第一个数字:";
    read num1;
    echo "请输入第二个数字:";
    read num2;
    echo "输入的两个数字分别是:${num1},${num2}";
    return $(($num1+$num2));
}
funWithReturn
echo "输入的两个数字之和是:$? ";


# 有参无返回
funWithParam(){
    echo "第一个参数为:$1";
    echo "第二个参数为:$2";
    echo "第三个参数为:$3";
    echo "第十一个参数为:$11";
    echo "第十一个参数为:${11}";
    echo "总计参数的个数:$#";
    echo "作为字符串输出所有参数:$*";
}
funWithParam 1 12 34 45 56 1 23 12 1 a b c e


# 有参有返回
funWithParam(){
    echo "第一个参数为:$1";
    echo "第二个参数为:$2";
    echo "第三个参数为:$3";
    echo "第十一个参数为:$11";
    echo "第十一个参数为:${11}";
    echo "总计参数的个数:$#";
    echo "作为字符串输出所有参数:$*";

    sum=0;
    for i in $*
    do
        sum=`expr ${sum} + ${i}`;
    done
    return ${sum};
}
funWithParam 1 12 34 45 56 1 23 12 1 12 1 2 3 4
echo "参数的和为:$?"

 

2、文件包含

  和其他语言一样,Shell 也可以包含外部脚本。这样可以很方便的封装一些公用的代码作为一个独立的文件。可以引入外部文件的一个函数,当然也可以引入外部文件的一个变量。

格式

格式一:. filename   # 注意点号(.)和文件名中间有一空格

格式二:source filename

 

eg:计算两个数的加减乘除

constant.sh

#!/usr/bin/env bash

# 常量 readonly ZERO
=0;

add.sh

#!/usr/bin/env bash

# desc:加法
function add(){
    return `expr $1 + $2`;
}

subtract.sh

#!/usr/bin/env bash

# desc:减法
function subtract1(){
    return `expr $1 - $2`;
}

function subtract2(){
   result_subtract=`expr $1 - $2`;
}

multiplication.sh

#!/usr/bin/env bash

# desc:乘法
function multiplication(){
    return `expr $1 \* $2`;
}

division.sh

#!/usr/bin/env bash

# desc:除法
function division(){
    if [ $2 == 0 ]
    then
        echo "除数不能为0";
        break;
    else
        return `expr $1 / $2`;
    fi
}

main.sh

#!/usr/bin/env bash

# 多种引入外部文件的方式 source .
/constant.sh . ./add.sh . ./subtract.sh . ./multiplication.sh . ./division.sh # 两个数的加、减、乘、除计算 function execute(){ echo "输入的参数为:$*"; add $1 $2 echo "这两个数的和为:$?"; subtract1 $1 $2 echo "这两个数的差为(方法一):$?"; subtract2 $1 $2 echo "这两个数的差为(方法二):${result_subtract}"; if [ $result_subtract -eq $ZERO ] then echo "两数相等"; else echo "两数不等"; fi multiplication $1 $2 echo "这两个数的积为:$?"; division $1 $2 echo "这两个数的商为:$?"; }
# 函数调用 execute
6 12 # 输出 输入的参数为:6 12 这两个数的和为:18 这两个数的差为(方法一):250 这两个数的差为(方法二):-6 两数不等 这两个数的积为:72 这两个数的商为:0

  这里有个坑,不知大家注意到了没。在计算两个数的减法时,我写了两个方法,方法一直接返回,方法二采用了变量赋值。可以看到,这两个方法的计算结果中,方法二的计算结果是正确的。为啥是这样呢,还记得之前我们强调的吗——Shell函数的return只能返回0~255范围内的数。如果要返回超出此范围的数,就要使用全局变量。

 

3、输入输出重定向

  大多数 UNIX 系统命令从你的终端接受输入并将所产生的输出发送回​​到您的终端。一个命令通常从一个叫标准输入的地方读取输入,默认情况下,这恰好是你的终端。同样,一个命令通常将其输出写入到标准输出,默认情况下,这也是你的终端。

  3.0、输入输出流

一般情况下,每个 Unix/Linux 命令运行时都会打开三个文件:

·标准输入文件(stdin):stdin的文件描述符为0,Unix程序默认从stdin读取数据。

·标准输出文件(stdout):stdout 的文件描述符为1,Unix程序默认向stdout输出数据。

·标准错误文件(stderr):stderr的文件描述符为2,Unix程序会向stderr流中写入错误信息。

默认情况下,command > file 将 stdout 重定向到 file,command < file 将stdin 重定向到 file。

3.1、输出重定向

  没啥好说的。就是把原本输出到控制台的内容输出到了文件里面。

  命令格式:command > file  (覆盖式)或 command >> file (追加式)

#!/usr/bin/env bash

# eg:输出到test.txt,会进行覆盖
echo "这是我的第一个shell输出" > test.txt
echo "这是我的第二个shell输出" > test.txt


# eg:输出到test.txt,不会进行覆盖
echo "这是我的第三个shell输出" >> test1.txt
echo "这是我的第四个shell输出" >> test1.txt

test.txt

这是我的第二个shell输出

test1.txt

这是我的第三个shell输出
这是我的第四个shell输出

3.2、输入重定向

      原先咱们是从控制台上输入,比如你用read命令读取你的输入,而“输入重定向”就是从文件里面读取,当做输入。

      命令格式:command < file

    eg:

# wc -l 用于统计行数,具体用法可以看本文最后
# 假设users文件有2行
wc -l users

# 输出
2 users


wc -l < users

# 输出
2

# 两个例子的结果不同:第一个例子,会输出文件名;第二个不会,因为它仅仅知道从标准输入读取内容。

命令格式:

# 执行command1,从文件infile读取内容,然后将输出写入到outfile中。
command1 < infile > outfile

默认情况下,command > file 将 stdout 重定向到 file,command < file 将stdin 重定向到 file。

如果希望 stderr 重定向到 file,可以这样写:

$ command 2 > file

如果希望 stderr 追加到 file 文件末尾,可以这样写:

$ command 2 >> file

2 表示标准错误文件(stderr)。如果希望将 stdout 和 stderr 合并后重定向到 file,可以这样写:

$ command > file 2>&1

或者
$ command >> file 2>&1

如果希望对 stdin 和 stdout 都重定向,可以这样写:

$ command < file1 >file2

3.3、Here Document

  Here Document 是 Shell 中的一种特殊的重定向方式,用来将输入重定向到一个交互式 Shell 脚本或程序。它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command。

基本格式:

command << delimiter
    document
delimiter

注意:

  1)结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。

  2)开始的delimiter前后的空格会被忽略掉。

eg:

wc -l << EOF
    欢迎来到
    菜鸟教程
    www.runoob.com
EOF

# 输出
3

#########################
cat << EOF
    欢迎来到
    我的博客
EOF

输出:
欢迎来到
我的博客

3.4、/dev/null文件

  如果希望执行某个命令,但又不希望在屏幕上显示输出结果,那么可以将输出重定向到 /dev/null。/dev/null 是一个特殊的文件,写入到它的内容都会被丢弃;如果尝试从该文件读取内容,那么什么也读不到。但是 /dev/null 文件非常有用,将命令的输出重定向到它,会起到"禁止输出"的效果。

格式:

command > /dev/null

  如果希望屏蔽 stdout 和 stderr,可以这样写:command > /dev/null 2>&1             (这里的 2> 之间不可以有空格,2> 是一体的时候才表示错误输出);

 

4、其他

  4.1、Linux wc命令

    wc用于计算字数。利用wc指令我们可以计算文件的Byte数、字数、或是列数,若不指定文件名称、或是所给予的文件名为"-",则wc指令会从标准输入设备读取数据。

    参数:

    

wc testfile                                # testfile文件的统计信息  
3 92 598 testfile                          # testfile文件的行数为3、单词数92、字节数598 


wc testfile testfile_1 testfile_2          #统计三个文件的信息  
3 92 598 testfile                          #第一个文件行数为3、单词数92、字节数598  
9 18 78 testfile_1                         #第二个文件的行数为9、单词数18、字节数78  
3 6 32 testfile_2                          #第三个文件的行数为3、单词数6、字节数32  
15 116 708 总用量                           #三个文件总共的行数为15、单词数116、字节数708

 

 

参考:https://www.runoob.com/linux/linux-shell.html

 

posted @ 2020-08-18 18:17  Erneste  阅读(190)  评论(0编辑  收藏  举报