草庐IT

尝试阅读理解一份linux shell脚本

ENG八戒 2023-03-28 原文

以下内容为本人的学习笔记,如需要转载,请声明原文链接 微信公众号「englyf」https://mp.weixin.qq.com/s/YkJsqlqsviWtHHArONv3Rw


从头一二去阅读语法和命令说明,对于脚本小白来说比较枯燥,难以坚持,所以这里选择对一份完整的shell脚本代码来逐行逐段解读,希望可以一渡小白,帮助我们快速进入脚本的大门_
司机要开车了:


#!/bin/sh

用注释的形式说明文件打开类型,此处意指本脚本需要用 /bin/sh 打开。

#V1.0 Added hardware type detection
#V1.1 xxxx-xx-xx 重构脚本
#V1.2 xxxx-xx-xx 将日志信息输出到U盘
#V1.3 xxxx-xx-xx 读取存储的检测结果值
#V1.4 xxxx-xx-xx 添加校验程序版本,U盘下有对应版本信息文件才校验

上面几行表示注释内容。单行注释,以#开始

UDiskMountDir=$(df | grep /mnt/sd | awk 'END{print $NF}')

声明变量 UDiskMountDir 并赋值。
$() 括号内填入命令并执行,最后返回输出到变量 UDiskMountDir。

df命令显示liinux系统上的文件系统磁盘使用情况统计。后边还可以带选项,执行更复杂的输出。命令后边加 '|' 表示将此命令的输出作为后边紧接着的命令的输入。

grep命令用于对文本按行搜索然后输出该行。英文全称 global search regular expression(RE) and print out the line。grep /mnt/sd 表示搜索输入信息里包含 /mnt/sd的一行,并输出该行。

awk命令用于对文本处理,END表示执行最后的运算或者打印最终的输出结果,这里用于打印输出,$NF 表示列数,print $NF 表示打印最后一个字段,各个字段默认以空格划分,可用选项 -F指定使用什么字符串来划分。

LogFile=${UDiskMountDir}/Debug.log

声明变量 LogFile 并以右边的内容赋值。将变量 UDiskMountDir 所代表的路径名与后边的字符串结合为新的文件名。一般命令中用到的文件名,要不是带相对路径的文件名,如 ./diretion/filename,就是带绝对路径的文件名,如 /root/diretion/filename

${UDiskMountDir} 表示引用变量 UDiskMountDir 的值。

AllCheckPassed=true

声明变量 AllCheckPassed 并赋值为true。

AddError()
{
    AllCheckPassed=false
    echo "-200" > /tmp/VerCheckRes
}

声明函数 AddError,输入参数无需声明。

变量 AllCheckPassed 赋值为true。

echo命令用于输出字符串。echo "-200" > /tmp/VerCheckRes 将 -200 输出到文件 /tmp/VerCheckRes 并覆盖原有内容。如果将符号 > 换成 >> 则是将内容追加到最后位置。

AddRecord()
{
    if [ ! -d "/userdata/recordsDir/" ];then
        mkdir -p /userdata/recordsDir/
    fi
    echo "$*" > /userdata/recordsDir/test
    sync
    AddDebugLog2UDisk $*
}

声明函数 AddRecord。

if 表示 if 语句的开始。if[]方括号内填判断条件,如为 true 则执行 then 后的语句,否则跳出 if 语句。fi表示 if 语句的结束。和 C 语言类似,if 语句也可以有 else 甚至 else if (shell 内应该写成 elif)。if/elifthen 如果不在同一行则可以省略符号 ;if [ ! -d "/userdata/recordsDir/" ] 判断目录 /userdata/recordsDir/ 是否存在,并对结果值取反(!)。

mkdir命令用于构建目录。带选项 -p表示构建路径下所有的目录。mkdir -p /userdata/recordsDir/创建目录 /userdata/recordsDir/, /userdata/ 如果不存在也会被构建。

echo "$*" > /userdata/recordsDir/test 将函数输入的所有参数都输出到文件 '/userdata/recordsDir/test' 并覆盖原有内容。'$*' 表示当前函数或者脚本的所有输入参数,由于在函数内引用,所以这里表示函数的所有输入参数。

sync命令用于数据同步。Linux 系统中写入硬盘的数据往往会被先存放于 buffer 中,这样是为了效率起见,但是如果系统突然断电,那么数据就会丢失,这时可以调用 sync 将 buffer 中的数据同步到硬盘。

AddDebugLog2UDisk $* 调用函数 AddDebugLog2UDisk 并传入当前函数(AddRecord)的所有参数。

AddDebugLog2UDisk()
{
    echo "$*" >>"$LogFile"
    sync
}

声明函数 AddDebugLog2UDisk。

echo "\$*" >>"$LogFile" 表示将函数的所有输入参数输出到变量 LogFile 表示的文件中,以追加的形式输入到文件末尾。

sync同步数据到硬盘。

#生成下/tmp/App/version.txt版本信息

注释内容

rm -f "$LogFile"

rm命令用于移除文件。选项 -f表示强制。rm -f "$LogFile"强制移除变量 LogFile 代表的文件,变量 LogFile 存储的是文件名,包括路径。

echo "0" > /tmp/VerCheckRes

输出字符串 '0' 到文件 /tmp/VerCheckRes

sh /data/bin/run_normal.sh

在当前的环境中使用另一个shell来执行脚本文件 /data/bin/run_normal.sh,当前环境中变量的值可以在新脚本(/data/bin/run_normal.sh)中被使用(非引用,如果使用 source则变为引用),新脚本文件属性可以无执行权限。sh后边可以带选项 -n-x等,-n 用于进行shell脚本的语法检查,-x用于实现shell脚本的逐句跟踪调试并打印该行命令和状态等。

sleep 2

阻塞当前进程,睡眠 2 秒,和 sleep 2s 同样效果。sleep 2h 表示睡眠 2 小时。

sh /tmp/App/kill.sh

执行脚本文件 /tmp/App/kill.sh

killall MachineCheckNode

杀掉进程 MachineCheckNode,MachineCheckNode 为进程名。

lastSNScanResult=$(cat /userdata/recordsDir/scan)

读取文件 /userdata/recordsDir/scan 内容并赋值给变量 lastSNScanResult。

cat命令用于连接文件并输出内容到标准输出。

AddDebugLog2UDisk "last sn scan result:$lastSNScanResult"

调用函数 AddDebugLog2UDisk,并传递字符串 last sn scan result:$lastSNScanResult$lastSNScanResult 获取变量 lastSNScanResult 的值然后和前面的字符串结合成新的字符串。

#检查SN和markData
if [ ! -f /data/bin/sysData/sn ]; then 
    AddRecord "sn missing"
    AddError
else
    snContent=$(cat /data/bin/sysData/sn)
    AddRecord "sn is:$snContent"
    snLen=$(cat /data/bin/sysData/sn | awk '{print length($0)}')
    if [ "$snLen" != "16" ] ; then
        AddRecord "sn length error $snLen"
        AddError
    fi
fi

if [ ! -f /data/bin/sysData/sn ] 判断文件 /data/bin/sysData/sn 是否是常规文件而且存在,'!' 表示对结果值取反。

如果文件 /data/bin/sysData/sn 找不到或者不是常规文件,则调用后边的语句块,直到 else 为止。调用函数 AddRecord(传入字符串 "sn missing") 和 AddError(无参数传入)。

否则,获取文件内容并赋值给变量 snContent,调用函数 AddRecord(输入 sn is:$snContent", 字符串其中会插入变量 snContent 的值);声明变量 snLen,赋值为命令 cat /data/bin/sysData/sn | awk '{print length($0)}'的执行结果;判断 snLen 是否不等于 16,是则调用函数 AddRecord 和 AddError。

$(cat /data/bin/sysData/sn | awk '{print length($0)}') 读取文件 /data/bin/sysData/sn 的内容,通过管道('|')输入到后一命令语句,awk 计算输入内容的第一字段的字符长度并输出。

if [ ! -f /data/bin/sysData/markData ]; then 
    AddRecord "markData missing"
    AddError
fi

判断文件 /data/bin/sysData/markData 是否不存在,不存在则调用后边的语句(以 fi 为止)。

#检验版本信息
while read -r line || [ -n "${line}" ]; do
    [ "$line" = "" ] && continue
    item_filename=$(echo "$line" | awk -F":" '{print $1}')
    item_filemd5=$(echo "$line" | awk -F":" '{print $2}')
    if [ ! -e "$item_filename" ]; then
        AddRecord "$item_filename missing"
        AddError
    fi
    acture_md5=$(md5sum "$item_filename" | awk '{print $1}')
    if [ "$item_filemd5" = "$acture_md5" ]; then
        continue
    else
        AddRecord "$item_filename" "md5 diff! file:" "$item_filemd5" "acture:" "$acture_md5"
        AddError
    fi
done </tmp/MachineDecDir/check_md5list

while read xxx; do
...
done </filename
循环读取文件名 filename 所指向的文件的内容,每读取一行存入 xxx 变量。read xxx为 while 语句的判断语句,判断语句和 do如果不在同一行,则 ;可省略,如:
while read xxx
do
...
done </filename

read 后边的 -r 表示读取内容过程中对特殊字符有效,如 / (输入未结束需要换行继续输入的特殊符号),'/n' 等等。

[ -n "${line}" ] 判断 line 的内容不为空。而前边的 '||' 表示前一语句 read -r line 返回 false 才执行这个语句。

[ "$line" = "" ] && continue 表示如果 line 的内容为空则执行 continue;continue 命令用于跳过当前循环内容回去执行 while 判断语句。

 item_filename=$(echo "$line" | awk -F":" '{print $1}') 声明变量 item_filename 并以后边语句执行结果赋值。awk -F":" '{print $1}') 将输入的内容按照 : 来划分字段并输出第1个字段。awk -F":" '{print $2}') 则是将输入的内容按照 ':' 来划分字段并输出第2个字段。

if [ ! -e "$item_filename" ] 用于判断变量 item_filename 代表的文件名所指的文件是否不存在。!表示取反。if 判断的内容为 true 则执行 then 后边的语句,直到 fi 为止,否则跳过。

md5sum "$item_filename" 表示计算变量 item_filename 代表的文件名所指的文件的MD5值。

 | awk '{print $1}' 将前一语句的输出通过管道连接到后边的这个语句的输入,默认按照空格划分内容并输出第1个字段内容。

if [ "$item_filemd5" = "$acture_md5" ]; then 判断 item_filemd5 和 acture_md5 所代表的内容作为字符串是否一样,是则执行 then 后的语句。continue 指示程序执行流程直接回到执行 while 判断语句。else 后的语句块当 if 的判断条件不为 true 时执行。

#找到 U 盘路径
UDiskMountDir=$(df | grep /mnt/sd | awk 'END{print $NF}')
#校验mcu程序版本,U盘下有mcuversion.txt才校验
UDiskMountDirfile="${UDiskMountDir}/SpecialDir/mcuversion.txt"
CurMcuVersionFile="/tmp/mcuversion.txt"
if [ -f "$UDiskMountDirfile" ]; then
    TargetMcuVersion=$(cat "$UDiskMountDirfile")
    if [ -f "$CurMcuVersionFile" ]; then
        CurMcuVersion="$(cat "$CurMcuVersionFile")"
        if [ "$TargetMcuVersion" != "$CurMcuVersion" ]; then
            AddRecord "mcuversion diff cur:$CurMcuVersion target:$TargetMcuVersion"
            AddError
        fi
    else
        AddRecord "$CurMcuVersionFile not found"
        AddError
    fi
fi

UDiskMountDirfile="${UDiskMountDir}/SpecialDir/mcuversion.txt" 声明变量 UDiskMountDirfile 并赋值为其后的内容,其后是将变量 UDiskMountDir 所代表的路径名与后边的字符串结合为新的文件名。

if 语句可以有多重嵌套,如上。

[ "$TargetMcuVersion" != "$CurMcuVersion" ] 判断 TargetMcuVersion 和 CurMcuVersion 所代表的内容是否不相等。

#写入检查结果
if [ $AllCheckPassed = "true" ]; then
    AddRecord "Pass"
else
    AddRecord "Fail"
fi

if [ $AllCheckPassed = "true" ]; then 判断 AllCheckPassed 的值是否等于字符串 "true",是则调用函数 AddRecord 并传入参数字符串 "Pass",否则调用函数 AddRecord 并传入参数字符串 "Fail"。

#启动checkall
/data/bin/Factory/MachineCheckNode MachineImcomingTest

调用绝对路径下的程序 MachineCheckNode,并传入字符串参数 MachineImcomingTest。此种调用方式,要求程序文件 MachineCheckNode 具有可执行的权限属性x。想要查看指定路径下所有文件或某个文件的属性可以使用命令 ls -l查看,r表示可读,w表示可写,x表示可执行。

username@DESKTOP-ABCDEF:/mnt/d/username/work/temp$ ls -l
total 0
drwxrwxrwx 1 username username 4096 May 21 16:10 Udisk_IQC

这篇讲解到此为止,下期再见!

有关尝试阅读理解一份linux shell脚本的更多相关文章

  1. ruby - ECONNRESET (Whois::ConnectionError) - 尝试在 Ruby 中查询 Whois 时出错 - 2

    我正在用Ruby编写一个简单的程序来检查域列表是否被占用。基本上它循环遍历列表,并使用以下函数进行检查。require'rubygems'require'whois'defcheck_domain(domain)c=Whois::Client.newc.query("google.com").available?end程序不断出错(即使我在google.com中进行硬编码),并打印以下消息。鉴于该程序非常简单,我已经没有什么想法了-有什么建议吗?/Library/Ruby/Gems/1.8/gems/whois-2.0.2/lib/whois/server/adapters/base.

  2. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  3. ruby-on-rails - 独立 ruby​​ 脚本的配置文件 - 2

    我有一个在Linux服务器上运行的ruby​​脚本。它不使用rails或任何东西。它基本上是一个命令行ruby​​脚本,可以像这样传递参数:./ruby_script.rbarg1arg2如何将参数抽象到配置文件(例如yaml文件或其他文件)中?您能否举例说明如何做到这一点?提前谢谢你。 最佳答案 首先,您可以运行一个写入YAML配置文件的独立脚本:require"yaml"File.write("path_to_yaml_file",[arg1,arg2].to_yaml)然后,在您的应用中阅读它:require"yaml"arg

  4. ruby-on-rails - 每次我尝试部署时,我都会得到 - (gcloud.preview.app.deploy) 错误响应 : [4] DEADLINE_EXCEEDED - 2

    我是Google云的新手,我正在尝试对其进行首次部署。我的第一个部署是RubyonRails项目。我基本上是在关注thisguideinthegoogleclouddocumentation.唯一的区别是我使用的是我自己的项目,而不是他们提供的“helloworld”项目。这是我的app.yaml文件runtime:customvm:trueentrypoint:bundleexecrackup-p8080-Eproductionconfig.ruresources:cpu:0.5memory_gb:1.3disk_size_gb:10当我转到我的项目目录并运行gcloudprevie

  5. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

  6. UE4 源码阅读:从引擎启动到Receive Begin Play - 2

    一、引擎主循环UE版本:4.27一、引擎主循环的位置:Launch.cpp:GuardedMain函数二、、GuardedMain函数执行逻辑:1、EnginePreInit:加载大多数模块int32ErrorLevel=EnginePreInit(CmdLine);PreInit模块加载顺序:模块加载过程:(1)注册模块中定义的UObject,同时为每个类构造一个类默认对象(CDO,记录类的默认状态,作为模板用于子类实例创建)(2)调用模块的StartUpModule方法2、FEngineLoop::Init()1、检查Engine的配置文件找出使用了哪一个GameEngine类(UGame

  7. postman——集合——执行集合——测试脚本——pm对象简单示例02 - 2

    //1.验证返回状态码是否是200pm.test("Statuscodeis200",function(){pm.response.to.have.status(200);});//2.验证返回body内是否含有某个值pm.test("Bodymatchesstring",function(){pm.expect(pm.response.text()).to.include("string_you_want_to_search");});//3.验证某个返回值是否是100pm.test("Yourtestname",function(){varjsonData=pm.response.json

  8. CAN协议的学习与理解 - 2

    最近在学习CAN,记录一下,也供大家参考交流。推荐几个我觉得很好的CAN学习,本文也是在看了他们的好文之后做的笔记首先是瑞萨的CAN入门,真的通透;秀!靠这篇我竟然2天理解了CAN协议!实战STM32F4CAN!原文链接:https://blog.csdn.net/XiaoXiaoPengBo/article/details/116206252CAN详解(小白教程)原文链接:https://blog.csdn.net/xwwwj/article/details/105372234一篇易懂的CAN通讯协议指南1一篇易懂的CAN通讯协议指南1-知乎(zhihu.com)视频推荐CAN总线个人知识总

  9. TimeSformer:抛弃CNN的Transformer视频理解框架 - 2

    Transformers开始在视频识别领域的“猪突猛进”,各种改进和魔改层出不穷。由此作者将开启VideoTransformer系列的讲解,本篇主要介绍了FBAI团队的TimeSformer,这也是第一篇使用纯Transformer结构在视频识别上的文章。如果觉得有用,就请点赞、收藏、关注!paper:https://arxiv.org/abs/2102.05095code(offical):https://github.com/facebookresearch/TimeSformeraccept:ICML2021author:FacebookAI一、前言Transformers(VIT)在图

  10. ruby - 确定 ruby​​ 脚本是否已经在运行 - 2

    有没有一种简单的方法可以判断ruby​​脚本是否已经在运行,然后适本地处理它?例如:我有一个名为really_long_script.rb的脚本。我让它每5分钟运行一次。当它运行时,我想看看之前运行的是否还在运行,然后停止第二个脚本的执行。有什么想法吗? 最佳答案 ps是一种非常糟糕的方法,并且可能会出现竞争条件。传统的Unix/Linux方法是将PID写入文件(通常在/var/run中)并在启动时检查该文件是否存在。例如pid文件位于/var/run/myscript.pid然后你会在运行程序之前检查它是否存在。有一些技巧可以避免

随机推荐