bash编程(马哥)

news/2024/12/27 4:35:28/

bash基础特性:

    命令行展开:~,{}

    命令别名:alias,unalias

    命令历史:history

    命令和路径补全:$PATH

    glob通配符:*,?,[],[^],

    快捷键:Ctrl+{a,e,l,c,u,k}

命令bash:

bash通配符及特殊符号:

通配符:

    ?:任意一个字符;

    *:匹配任意个任意字符;

    []:匹配括号内的任意一个字符;

    [^]:匹配非括号内的任意一个字符

特殊字符:

    '':单引号中,所有特殊字符都没有特殊含义;

    "":双引号中,除$(调用变量的值)、`(命令引用)、\(转义符)外,其他特殊字符没有特殊含义;

    ` `:等同于$();达到命令引用,即先执行本处命令;

    \:转义符,使跟\之后的特殊字符失去特殊含义;

    $:调用变量的值;

    #:在shell脚本中,#开头的行代表注释

bash脚本:

运行脚本:

    1、给予执行权限,通过具体的文件路径指定文件执行;  

    2、直接运行解释器,将脚本作为解释器程序的参数运行;

数据类型:

    一、字符

    二、数值

        1、整型

        2、浮点型

        (1)、单精度浮点型

        (2)、双精度浮点型

        3、布尔型

bash中的算术运算:man let

    +,-,*,/,%

实现算术运算:(乘法符号有些场景中需转义\)

    1、let var=算术表达式;例如 let sum=$num1+$num2

    2、var=$[算术表达式];例如 echo $[$num1+$num2]

    3、var=$((算术表达式));例如echo $(($num1+$num2))

    4、var=$(expr arg1 arg2 arg3 …);例如sum=$(expr $num1 + $num2)

bash内建的随机数生成器:$RANDOM;

    例如取出0-60之间# echo $[$RANDOM%60+1]

增强型赋值:man let;自增、自减:

    +=,-=,*=,/=,%=

    例如let count+=1、let count++

条件测试:判断某需求是否满足,需要有测试机制来实现;

测试命令:

    test EXPRESSION

    [ EXPRESSION ]

    [[ EXPRESSION ]]



 

bash的测试类型:

一、数值测试:

    -gt:大于;

    -lt:小于;

    -ge:大于等于;

    -le:小于等于;

    -eq:等于;

二、字符串测试:(注意:用于字符串比较的操作数要使用引号)

    ==:等于;

    >:大于;

    <:小于;

    !=:不等于;

    =~:左侧字符串是否被右侧pattern所匹配;此表达式一般用于[[ ]]中;

    -z “string”:字符串是否为空,空则为真,不空则为假;

    -n “string”:字符串是否非空,非空为真,空则为假;

三、文件测试:(man bash)

    -e file:存在测试;

    -b file:存在且为块设备文件;

    -c file:存在且为字符设备文件;

    -d file:存在且为目录文件;

    -f  file:存在且为普通文件;

    -g file:存在且拥有sgid权限;

    -u file:存在且拥有suid权限;

    -k file:存在且拥有sticky权限;

    -r file:存在且可读;

    -w file:存在且可写;

    -x file:存在且可执行;

    -s file:存在且非空;

    -t fd:fd表示文件描述符是否已经打开且与某终端相关;

    file1 -ef file2:是否为同一个设备上的相同inode;

    file1 -nt file2:file1是否新于file2;

    file1 -ot file2:file1是否旧于file2;

组合测试条件:

第一种方式:

command1 && command2

command1 || command2

! command

例如:[ -z “$(hostname)” ] || [ “$(hostname)” == “localhost.localdomain” ]

第二种方式:

expression1 -a expression2

expression1 -o expression2

! expression

例如:[ -z “$(hostname)” -o “$(hostname)” == “localhost.localdomain” ]

bash自定义退出状态码:

exit [n]:自定义退出状态码;

注意:脚本中一旦遇到exit命令,脚本会立即终止;终止状态码取决于exit命令后的数字;

注意:如果未给脚本指定状态退出码,整个脚本的退出状态码取决于脚本中执行的最后一条命令的状态码;


 

bash编程之用户交互:

read -p “提示语” 变量名

bash脚本编程:

    整数值比较:-ge 大于等于; -gt 大于; -eq 相等; -lt 小于; -ne 不相等;-le小于等于

    顺序执行

    选择执行:if、case

    循环执行:for、while、until

    函数:结构化编程及代码重用;function

选择执行:if语句(逐条件进行判断;第一次遇到为“真”条件时,执行其分支,而后结束)

单分支:

    if 判断条件;then

        条件为真的分支代码

    fi

双分支:

    if 判断条件;then

        条件为真的分支代码

    else

        条件为假的分支代码

    fi

多分支:

    if 判断条件1;then

        条件为真的分支代码

    elif 判断条件2;then

        条件为真的分支代码

    elif 判断条件3;then

        条件为真的分支代码

        …  

    else

        条件为假的分支代码

    fi

case语句:

    case $变量名 in

    模式1)

        命令序列1

        ;;

    模式2)

        命令序列2

        ;;

    *)

        默认执行的命令序列

        ;;

    esac

bash编程之for语句

循环执行:for语句

for 变量 in 列表;do

    循环体

done

for循环的特殊格式:

for ((控制变量初始化;条件判断表达式;控制变量的修正表达式));do

    循环体

done

控制变量初始化:仅在运行到循环代码段时执行一次;

控制变量的修正表达式:每轮循环结束会先进行控制变量修正运算,而后再做条件判断;

列表生成方式:

(1)直接给出列表;例如 for i in 1 2 3 4;then…

(2)整数列表:

(a){start..end};例如{1..10}代表1-10;

(b)$(seq [start [step]] end);例如:$(seq 1 2 10),代表10以内的奇数;

(3)返回列表的命令:$(COMMAND);例如$(ls /var)代表对列出/var目录下的文件列表;

(4)glob通配符

while语句:当条件为true时进入循环,条件为false时退出循环;

    while 条件;do

        循环体

    done

while循环的特殊用法(遍历文件的每一行):依次读取/path/to/somefile文件中的每一行,且将行赋值给变量line;

    while read line;do

        循环体

    done < /path/to/somefile

until语句:当条件为false时进入循环,条件为true时退出循环;

    until 条件;do

        循环体

    done

创建死循环:

    while true;do

        循环体

    done

    until false;do

        循环体

    done

循环控制语句(用于循环体)

continue [n]:提前结束第n层的本轮循环,而直接进入下一轮判断;

break [n]:提前结束循环;

function函数

语法一:

    function f_name {

        函数体...

    }

语法二:

    f_name () {

        函数体...

    }

bash脚本编程:

    变量:存储单个元素的内存空间;

    数组:存储多个元素的连续的内存空间;

数组名

    索引:编号从0开始,属于数值索引;注意:索引也可以使用自定义的格式,而不仅仅是数值格式;

bash的数组支持稀疏格式;

    引用数组中的元素:${ARRAY_NAME[INDEX]}

数组长度(数组中元素的个数):

    ${#ARRAY_NAME[*]}、${#ARRAY_NAME[@]}

引用所有元素

    ${ARRAY_NAME[*]}、${ARRAY_NAME[@]}

取出特定元素:数组切片

    ${ARRAY_NAME[@]:offset:number}

offset:要跳过的元素个数

number:要取出的元素个数;去偏移量之后的所有元素:${ARRAY_NAME[@]:offset};

声明数组:

declare -a ARRAY_NAME

declare -A ARRAY_NAME:关联数组:

数组元素的赋值方式:

    (1)一次只赋值一个元素:

        ARRAY_NAME[INDEX]=VALUE

    (2)一次赋值全部元素:

        ARRAY_NAME=(“VALUE1” “VALUE2″ …)

    (3)只赋值特定元素:

        ARRAY_NAME=([0]=”VALUE1″ [3]=”VALUE3” …)

    (4)让用户输入元素

        read -a ARRAY

例子:随机生产10个数保存于数组中,找出最大和最小值;

    declare -a nums

    declare -i max=0

    declare -i min=0

for i in {0..9};do

    nums[$i]=$RANDOM

    echo ${nums[$i]}

    if [ $i -eq 0 ];then

        min=${nums[$i]}

        max=${nums[$i]}

    else

        if [ ${nums[$i]} -gt $max ];then

            max=${nums[$i]}

        fi

        if [ ${nums[$i]} -lt $min ];then

            min=${nums[$i]}

        fi

    fi

done

echo “max: $max”

echo “min: $min”

向数组中追加元素:ARRAY[${#ARRAY[*]}]

删除数组中的某元素:unset ARRAY[INDEX]

关联数组:

declare -a ARRAY_NAME

ARRAY_NAME=([index_name1]=”VALUE1″     [index_name2]=”VALUE2″    …)

bash的字符串处理工具:

字符串切片:

    ${var:offset:number}:

    offset:要跳过的字符串个数

    number:要取出的字符串个数;去偏移量之后的所有字符串:${ARRAY_NAME[@]:offset};

    取字符串的最右侧几个字符:${var:  -lengh};注意冒号后面必须有一空白字符;

基于模式去子串:

${var#*word}:其中word可以是指定的任意字符;功能:自左而右,查找var变量所存储的字符串中,第一次出现的word,删除字符串开头至第一次出现word字符之间的所有字符;

${var##*word}:其中word可以是指定的任意字符;功能:自左而右,查找var变量所存储的字符串中,最后一次出现的word,删除字符串开头至最后一次出现word字符之间的所有字符;

${var%word*}:其中word可以是指定的任意字符;功能:自右而左,查找var变量所存储的字符串中,第一次出现的word,删除字符串最后一个字符向左至第一次出现word字符之间的所有字符;

${var%word*}:其中word可以是指定的任意字符;功能:自右而左,查找var变量所存储的字符串中,第一次出现的word,删除字符串最后一个字符向左至最后一次出现word字符之间的所有字符;

例子:url=http://www.lewis.com:80

${url##*:}:80

${url%%:*}:http

查找替换:

${var/pattern/substi}:查找var所表示的字符串中,第一次被pattern所匹配到的字符串,以substi替换之;

${var//pattern/substi}:查找var所表示的字符串中,所有能被pattern所匹配到的字符串,以substi替换之;

${var/#pattern/substi}:查找var所表示的字符串中,行首被pattern所匹配到的字符串,以substi替换之;

${var/%pattern/substi}:查找var所表示的字符串中,行尾被pattern所匹配到的字符串,以substi替换之;

查找并删除:

${var/pattern}:查找var所表示的字符串中,第一次被pattern所匹配到的字符串,删除之。

${var//pattern}:查找var所表示的字符串中,所有被pattern所匹配到的字符串,删除之。

${var/#pattern}:查找var所表示的字符串中,行首被pattern所匹配到的字符串,删除之。

${var/%pattern}:查找var所表示的字符串中,行尾被pattern所匹配到的字符串,删除之。

字符大小写转换:

${var^^}:把var中的所有小写字母转换成大写;

${var,,}:把var中的所有大小字母转换成小写;

变量赋值:

${var:-value}:如果var为空或未设置,那么返回value;否则,则返回var的值;

${var:=value}:如果var为空或未设置,那么返回value,并将value赋值给var;否则,则返回var的值;

${var:+value}:如果var非空,则返回value;

${var:?error_info}:如果var为空或未设置,那么返回error_info;否则,则返回var的值;

为脚本程序使用配置文件:

    定义文本文件,每行定义“name=value”

    在脚本中source此文件即可

mktemp命令:创建临时文件或目录

-d:创建临时目录

–tmpdir:指明临时文件目录位置

例子:mktemp /tmp/test.XXX

install命令:功用类似cp命令

install [OPTION]… [-T] SOURCE DEST

install [OPTION]… SOURCE… DIRECTORY

install [OPTION]… -t DIRECTORY SOURCE…

install [OPTION]… -d DIRECTORY…

-m MODE:指明权限

-o OWNER:指明属主

-g GROUP:指明属组

ldd:查找某命令文件所依赖的共享库

ldd [OPTION]… FILE…

练习:写一个脚本

(1)提示用户输入一个可执行命令名称;

(2)获取此命令所依赖到的所有库文件列表;

(3)复制命令至某目录(例如/mnt/sysroot)下的对应路径下;

(4)复制此命令依赖到的所有库文件至目标目录下的对应路径下;

进一步:每次复制完成一个命令后,不要退出,而是提示用户键入新的要复制的命令,并重复完成完成上述功能:直到用户输入quit退出;

#!/bin/bash

#

ch_root=”/mnt/sysroot”

cmdcp() {

if which $1 &> /dev/null;then

    cmd_path=$(which –skip-alias $1)

    cmd_dir=$(dirname ${cmd_path})

    [ -d ${ch_root}${cmd_dir} ] || mkdir -p ${ch_root}${cmd_dir}

    [ -f ${ch_root}${cmd_path} ] || cp ${cmd_path} ${ch_root}${cmd_dir}

    return 0

else

    echo “command not found.”

    return 1

fi

}

libcp() {

    lib_list=$(ldd /usr/bin/bash | grep -o ‘/[^[:space:]]\+’)

    for i in $lib_list;do

        lib_dir=$(dirname $i)

        [ -d ${ch_root}${lib_dir} ] || mkdir -p ${ch_root}${lib_dir}

        [ -f ${ch_root}$i ] || cp $i ${ch_root}${lib_dir}

    done

}

read -p “please input a command:” command

until [ $command == “quit” ];do

    if cmdcp $command ;then

        libcp$command

        echo “job $command is finish”

        read -p “please input a command:” command

    else

        read -p “please input a command:” command

    fi

done

资料来源于马哥linux视频教程


http://www.ppmy.cn/news/334601.html

相关文章

笑死!俩美女路边直播拉路人点歌,结果拉来了黄仁勋,全程魔性互动

&#x1f447;&#x1f447;关注后回复 “进群” &#xff0c;拉你进程序员交流群&#x1f447;&#x1f447; 来源丨新智元 https://mp.weixin.qq.com/s/07xW49EQNf3VIXkUQ8C4Lw 新智元报道 编辑&#xff1a;David 【新智元导读】两位人美歌甜的宝岛妹子主播&#xff0c;街头…

【C语言】那些优秀代码里的骚操作(持续更新…)

【C语言】那些优秀代码里的骚操作&#xff08;持续更新…&#xff09; 1、联合体union的妙用2、#include的本质是什么&#xff1f;3、脱裤子放屁的do{ }while(0)4、一个成熟的代码要学会自己写函数5、…… 语言这个东西&#xff0c;其实没有奇技淫巧&#xff0c;凡是可以写出来…

算法17: 删除排序链表中的重复元素(升序链表去重)

一、需求 给定一个已排序的链表的头 head &#xff0c; 删除所有重复的元素&#xff0c;使每个元素只出现一次 。返回 已排序的链表 。 示例 1&#xff1a; 输入&#xff1a;head [1,1,2] 输出&#xff1a;[1,2] 示例 2&#xff1a; 输入&#xff1a;head [1,1,2,3,3] 输出…

Vue中如何进行数据筛选与搜索功能实现

Vue中如何进行数据筛选与搜索功能实现 在Vue应用中&#xff0c;数据筛选和搜索是常见的需求。本文将介绍如何在Vue中进行数据筛选和搜索功能的实现&#xff0c;包括基于原生JavaScript的筛选和搜索、基于Lodash库的筛选和搜索、以及基于Vue插件的筛选和搜索。 基于原生JavaScr…

Vue中如何进行图表绘制

Vue中如何进行图表绘制 数据可视化是Web应用中非常重要的一部分&#xff0c;其中图表绘制是其中的重要环节。Vue作为一款流行的前端框架&#xff0c;提供了很多优秀的图表库&#xff0c;以满足不同业务场景下的需求。本文将介绍如何在Vue中进行图表绘制&#xff0c;包括使用Vu…

【Deno】denon deno热重载框架

官网 https://deno.land/x/denon2.5.0 简介 denon是 deno 中 nodemon 的替代品。denon会监视文件的改变&#xff0c;并自动重新启动程序&#xff0c;重新编译更改的源代码。 安装 ⚠️ Make sure you are using deno version ^1.6.0 to install this executable. You can u…

离散数学编程作业:输出n元集合的所有划分或幂集元素

编程题目&#xff08;二选一&#xff09;&#xff1a; 打印输出n元&#xff08;n1,2,3,4,5,6&#xff09;集合的所有划分。打印输出n元&#xff08;n1,2,3,4,5,6&#xff09;集合的幂集中的所有元素。 编程内容及要求&#xff08;二选一&#xff09;&#xff1a; 编写程序&a…

【0基础自研记录】ESP32-CAM自制个人网络监控

目的&#xff1a;实现一个小型家庭监控 一、前期准备 1.硬件准备 esp32-acm烧录板烧录线 2.软件准备 Arduion IDE CH340串口驱动 下载地址如下 Arduion IDE:https://www.arduino.cc/en/software CH340串口驱动 链接&#xff1a;https://pan.baidu.com/s/1ri8dK7wW6KFz8rOPs…