shell脚本——循环语句、sed、函数、数组、免交互expect

目录

循环语句

for

while 与 until

sed

基本用法

sed脚本格式

函数

注意事项

定义函数和调用函数

脚本中函数的位置

查看函数

删除函数

函数返回值

函数的传参操作

使用函数文件

递归函数 

数组

声明数组

数组切片

免交互expect

定义

 基本命令


循环语句

for

for循环需要知道循环的次数

格式1:

for  变量   in   变量范围

do
执行的命令
done

例如:计算1到100的和

#!/bin/bash
sum=0                # 定义和的变量
for i in {1..100}        # 遍历1到100
do
sum=$[sum+i]        # 计算1到100的和
done
echo $sum               # 输出总和

 

 格式2: 

for (( 表达式1; 表达式2; 表达式3 )); do 命令; done

for (( 表达式1; 表达式2; 表达式3 ))

do 命令

done

 例如:计算1到100的和

#!/bin/bash
sum=0
for ((i=0;i<=100;i++))
do
sum=$[sum+i]
done
echo $sum

for循环嵌套

例如:打印99乘法表

#!/bin/bash
for i in {1..9}
do
for ((j=1;j<=$i;j++))
do
echo -en  "$i*$j=$[i*j]\t"
done
echo
done


 for循环例子:

例1:批量添加用户

#!/bin/bash
ulist=$(cat /opt/user.txt)
for uname in $ulist
do
   useradd $uname
   echo "123123" |passwd --stdin  $uname &>/dev/null
done


 例2:批量修改后缀名

#!/bin/bash
cd /data
for file in *
do
name=`echo $file |cut -d "." -f1`
mv $file   ${name}.bak
done

 

while 与 until

相对于for循环,while循环不需要知道循环次数,但是需要给循环结束的条件,否则就是死循环。

while与until的区别:

while
当命令判断为假时停止

until
当命令判断为真时停止

例如:计算1-100所有偶数的和

#!/bin/bash
i=0
sum=0
while [ $i -le 100 ]
do
let sum+=$i
let i+=2
done
echo $sum



例如:猜数字游戏

#!/bin/bash
p=`echo $[RANDOM%1000+1]`
t=0
while true
do
read -p "请输入你猜的数字1-1000:" num
let t++
if [ $num -eq $p ]
then echo "恭喜你猜中了,随机数字是$p"
echo "您一共猜了${t}次";exit 0
elif [ $num -gt $p ]
then echo "您猜的数字高了"
else echo "您猜的数字低了"
fi
done


双重循环及跳出循环

  • break跳出单个循环后面加数字2则代表跳出两层循环

  • continue终止某次循环中的命令,但是不会完全终止命令

continue终止本次循环,继续下一次循环

如:

break结束当前循环,不在执行当前的循环

如:

sed

Sed是从文件或管道中读取一行,处理一行,输出一行;再读取一行,再处理一行,再输出一行,直到最后一行。每当处理一行时,把当前处理的行存储在临时缓冲区中,称为模式空间(PatternSpace),接着用sed命令处理缓冲区中的内容,处理完成后,把缓冲区的内容送往屏幕。

基本用法

格式:

sed [option]... 'script;script;...' [input  file...]
     选项         自身脚本语法         支持标准输入管道


常用选项:
-n    不输出模式空间内容到屏幕,即不自动打印
-e    多点编辑[root@www data]#sed -n -e '/^r/p'  -e'/^b/p' /etc/passwd
-f     FILE 从指定文件中读取编辑脚本
-r, -E     使用扩展正则表达式
-i.bak     备份文件并原处编辑
-s     将多个文件视为独立文件,而不是单个连续的长文件流

 -n    关闭自动打印功能

-r, -E     使用扩展正则表达式

 -i   与   -i.bak        修改文件内容,与修改前备

sed脚本格式

格式:

sed   '地址+命令'   文件

所谓的地址就是一些参数,或规定的条件,根据这些参数和条件进行打印或修改等操作

1. 不给地址:对全文进行处理(比如行号)

2. 单地址:
   #(1,2,3...):指定的行,$:最后一行,需要结合选项和命令进行匹配

   /pattern/:被此处模式所能够匹配到的每一行,正则表达式

3. 地址范围:
   n,n    表示从n行到第n行,3,6 从第3行到第6行

   n,+m   表示从n行到n+m行,3,+4 表示从3行到第7行

   /pat1/,/pat2/    表示从第一个正则表达式和第二个正则表达式之间的行

   n,/pat/  表示从n号行为开始找到 pat为止 

   /pat/,n    表示找到pat后n行为止

4. 步进:~
     1~2 奇数行
     2~2 偶数行

命令
p 打印当前模式空间内容,追加到默认输出之后(要与选项-n一起使用)

I (大写的 i )        忽略大小写输出

d 删除模式空间匹配的行,并立即启用下一轮循环

a [\]text 在指定行后面追加文本,支持使用\n实现多行追加
i [\]text 在行前面插入文本
c [\]text 替换行为单行或多行文本

w file 保存模式匹配的行至指定文件

r file 读取指定文件的文本至模式空间中匹配到的行后

= 为模式空间中的行打印行号

! 模式空间中匹配行取反处理

 面试题:查看几点到几点之间的日志

[root@heitui opt]#sed -n '/15:41/,/15:42/p' log                #此处的log文件在opt下


搜索替代,类似于vim编辑器末行模式的s///g

s/pattern/string/修饰符 查找替换,支持使用其它分隔符,可以是其它形式:s@@@,s###
替换修饰符:
g 行内全局替换
p 显示替换成功的行
w   /PATH/FILE 将替换成功的行保存至文件中
I,i   忽略大小写

函数

 格式1:

function   函数名 {

        命令序列

}

格式2:

函数名(){

        命令序列

}

格式3:

function  函数名  (){

        命令序列

}

三种格式都是同样的效果,推荐使用格式2,格式2最为简单

注意事项

  1. 函数调用,直接使用函数名即可,函数名就相当于命令

  2. 同名函数 后一个生效

  3. 调用函数一定要先定义,在调用,否则会报错

  4. 函数在调用时,不会在乎每个函数定义的顺序

定义函数和调用函数

[root@heitui ~]#fun1 () { hostname;date; }        #定义函数[root@heitui ~]#fun1        #调用函数
heitui
2023年 08月 19日 星期六 22:10:37 CST

脚本中函数的位置

脚本中使用函数,一定要在调用函数之前定义函数,否则会报错

如下:错误示范

 

 正确示范:

查看函数

[root@heitui ~]#declare -F     # 查看函数列表

[root@heitui ~]#declare -f       # 查看函数具体的内容

删除函数

格式:unset    函数名

[root@heitui opt]#unset fun1

函数返回值

return表示退出函数并返回一个退出值,脚本中可以用$?变量表示该值

使用原则:

  1. 函数一结束就去返回值,应为$?变量只返回执行的最后一条命令的退出返回码

  2. 退出码必须是0-255,超出的值将为除以256取余

 

函数的传参操作

 在调用函数的时候,直接在命令后面加上参数

格式:函数名   参数1  参数2  ...参数n

 

函数变量的作用范围:

  • 函数在shell脚本中仅在当前的shell环境中有效
  • shell脚本中函数的变量默认全局有效
  • 将变量限定在函数内部使用local命令

使用函数文件

创建专门存放函数的文件


递归函数 

计算阶乘

脚本如下:

#!/bin/bash
#
fact() {
if [ $1 -eq 0 -o $1 -eq 1 ]
then
echo 1
else
echo $[$1*$(fact $[$1-1])]fi
}
fact $1


数组

 数组分为两种:

  • 普通数组        普通数组以数字为下标
  • 关联数组        关联数组,不以数字为下标,可以使用字符串或字符为下标

数组名和索引

  • 索引的编号从0开始,属于数值索引

  • 索引可支持使用自定义的格式,而不仅是数值格式,即为关联索引,bash 4.0版本之后开始支持

  • bash的数组支持稀疏格式(索引不连续)

声明数组

#普通数组可以不事先声明,直接使用
declare -a ARRAY_NAME
#关联数组必须先声明,再使用
declare -A ARRAY_NAME

  定义数组格式:

格式1:数组名=(value0 value1 value2 value3 ......)格式2:数组名=([0]=value [1]=value1 [2]=value2 ....)格式3:列表名="value0 value1 value2 value3 ...... "数组名=($列表名)格式4: 数组名[0]="value1"数组名[1]="value2"数组名[2]="value3"

查看普通数组下标:${!a[*]}

 ${a[*]} 与 ${a[@]}的区别

列出数组中有多少值  使用 ${#a[*]} 或者  ${#a[@]}

数组切片

格式:${ARRAY[@]:offset:number}
# offset #要跳过的元素个数
# number #要取出的元素个数# 取偏移量之后的所有元素 
{ARRAY[@]:offset}

免交互expect

定义

是建立在tcl(tool command language)语言基础上的一个工具,常被用于进行自动化控制和测试,解决shell脚本中交互的相关问题

expect默认是没有安装的,需要自己安装

[root@heitui ~]#yum install -y expect

expect使用格式:expect [选项] [ -c cmds ] [ [ -[f|b] ] cmdfile ] [ args ]

expect中相关命令

  • spawn 启动新的进程(监控,捕捉)

spawn    passwd   root

  • expect 从进程接收字符串

  • send 用于向进程发送字符串

  • exp_continue 匹配多个字符串在执行动作后加此命令

  • interact 允许用户交互expect eof

 基本命令

(1)脚本解释器

expect 脚本中首先引入文件,表明使用的事哪一种shell

#!/usr/bin/expect

(2)spawn

spawn 后面通常跟一个Linux执行命令,表示开启一个会话、进程,并跟踪后续交互信息

例: spawn passwd root

(3)expect

判断上次输出结果中是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回;只能捕捉有swpan启动的进程输出;

用于接受命令执行后的输出,然后和期望的字符串匹配

(4)send

向进程发送字符串,用于模拟用户的输入:该命令不能自动回车换行,一般要加 \r (回车) 或者\ n

例子方式一:
expect "密码" {send "abc123\r"}     #同一行send部分要有{}方式二:
expect "密码"  
send "abc123\r"                    # 换行send部分不需要有{}方式三:
expect 支持多个分支
expect          #只要匹配了其中一个情况,执行相应的send 语句后退出该expect 语句
只匹配一次
expect
{
{"密码1"  {send "abc123\r"}
{"密码2"  {send "123123\r"}
{"密码3"  {send "123456\r"}}

(5) 结束符

expect eof

表示交互结束,等待执行结束,退回到原用户,与spawn对应

比如切换到root用户,expect 脚本默认的等待时间是10s,当执行王命令后,默认停留10s后,自动切回原用户.

interact

执行完成后保持交互状态, 把控制权交给控制台,会停留在目标终端而不是退回到原终端

需要注意的是,expect eof 与 interact 只能二选一

(6)set

expect 默认的超时时间是10秒,通过set 命令可以设置会话超时时间,若不限制超时时间则应设置为-1

例子: set time out 30

(7) exp_continue

exp_continue 表示允许 expect 继续向下执行指令.

exp_continue附加于某个expect 判断选项之后,可以是该项被匹配后还能继续匹配expect 判断语句内的其他项。exp_continue类似于控制语句的continue语句。表示允许expect继续向下执行命令。

例如:

expect

{

"(yes/no)" {send "yes\r";exp_continue;}

"*password" {set timeout 300; send "abc123\r"}

}

注意:使用exp_continue时,如果跟踪像passwd这样输入密码后就结束进程的命令,expect {}外不要加上expect eof 因为spawn进程结束后悔默认向expect 发送eof,会导致后面的expect eof执行报错

(8)send_user

表示回显命令与echo相同

(9)接收参数(位置变量)

expect 脚本可以接受从bash命令行传递参数,使用 [lindex $argv n]获得。其中你从0开始,分别表示第一个,第二个,第三个.....参数

例子:

set hostname [lindex $argv 0] 相当于hostname=$1

set password [lindex $argv 1] 相当于passswd=$2

expect直接执行,需要expect命令去执行脚本

例子

#!/usr/bin/expect
spawn ssh 192.168.3.101   # ssh 远程登陆,使用spawn执行
expect {"yes/no" { send "yes\n";exp_continue }    # 捕捉yes/no,然后发送yes,\n起到回车的作用"password" { send  "123123\n" }    # 继续捕捉password,然后发送magedu,\n起到回车的作用
}       
interact    # 结束符,结束后不返回原用户,停留在远程用户端上

脚本中使用参数的方式

set   name   [lindex $argv 0]
set    pd    [lindex $argv 1]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/56254.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python3/pip3 SSL: CERTIFICATE_VERIFY_FAILED] certificate verify failed

环境&#xff1a; mac os 背景&#xff1a; 电脑之前安装的是python3.9 &#xff0c; 现在升级到python3.10。 从python官网下载macos版本的python3.10 pkg。 双击安装。 程序使用aiohttp访问ebay 。 出错&#xff1a; aiohttp.client_exceptions.ClientConnectorCertifi…

MySql015——使用子查询

一、创建customers表 ######################## # Create customers table ######################## use study;CREATE TABLE customers (cust_id int NOT NULL AUTO_INCREMENT,cust_name char(50) NOT NULL ,cust_address char(50) NULL ,cust_city char…

如何让qt tableView每个item中个别字用不同颜色显示?

如何让qt tableView每个item中个别字用不同颜色显示&#xff1f; 从上面图片可以看到&#xff0c;Item为红色&#xff0c;数字5为黑色。 要实现在一个控件实现不同颜色&#xff0c;目前想到的只有QTextEdit 、QLabel。有两种方法&#xff0c;第一种是代理&#xff0c;第二种是…

yolov5添加SimAM注意力机制(yolov7同理)

SimAM注意力机制简介 关于SIMAM注意力机制的原理这里不再详细解释,这篇发在Proceeddings of the 38th Internation Conference on Machine Learning.论文参考如下论文链接here   yolov5中添加SimAM注意力机制 注意力机制分为接收通道数和不接受通道数两种。这次属于不接受通…

数据库相关知识2

数据库知识2 关系完整性 数据完整性 指的是数据库中的数据的准确性和可靠性 实体完整性约束&#xff1a; 目的&#xff1a; 在表中至少有一个唯一的 标识&#xff0c;主属性字段中&#xff0c;不为空&#xff0c;不重复 主键约束&#xff1a;唯一 不重复 不为空 primary k…

c语言实现堆

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、树1、树的概念2、树的相关概念3、树的表示 二、二叉树1、二叉树概念2、特殊的二叉树3、二叉树的性质4、二叉树的顺序结构5、二叉树的链式结构 三、堆(二叉树…

C# textBox1.Text=““与textBox1.Clear()的区别

一、区别 textbox.Text "" 和 textbox.Clear() 都可以用于清空文本框的内容&#xff0c;但它们之间有一些细微的区别。 textbox.Text "": 这种方式会将文本框的 Text 属性直接设置为空字符串。这样会立即清除文本框的内容&#xff0c;并将文本框显示为空…

【leetcode 力扣刷题】双指针///原地扩充线性表

双指针///原地扩充线性表 剑指 Offer 05. 替换空格定义一个新字符串扩充字符串&#xff0c;原地替换思考 剑指 Offer 05. 替换空格 题目链接&#xff1a;剑指 Offer 05. 替换空格 题目内容&#xff1a; 这是一道简单题&#xff0c;理解题意&#xff0c;就是将字符串s中的空格…

拼多多开放平台的API接口可以获取拼多多电商数据。以下是API接口流程

使用拼多多开放平台的API接口可以获取拼多多电商数据。以下是一般的API接口流程&#xff1a; 1. 注册开发者账号&#xff1a;首先&#xff0c;您需要在拼多多开放平台注册一个开发者账号。通过开发者账号&#xff0c;您可以获得API密钥和其他必要的信息。 2. 鉴权与认证&…

数据结构1

数据结构是计算机科学中存储和组织数据的一种方式&#xff0c;它定义了数据的表示方式和对数据进行操作的方法&#xff0c;常见的数据结构包括数组、栈、链表、队列、树、图等。 目录 一、常见的数据结构 1.数组 2.栈 3.队列 4.链表 5.树 6.图 一、常见的数据结构 1.数…

自动设置服务器全教程

亲爱的爬虫探险家&#xff01;在网络爬虫的世界里&#xff0c;自动设置代理服务器是一个非常有用的技巧。今天&#xff0c;作为一家代理服务器供应商&#xff0c;我将为你呈上一份轻松实用的教程&#xff0c;帮助你轻松搞定爬虫自动设置代理服务器。 一、为什么需要自动设置代…

前端如何走通后端接口

0 写在前面 现在基本都是前后端分离的项目了&#xff0c;那么前端小伙伴如何获取后端小伙伴接口呢&#xff1f; 1 条件 同一WiFi下&#xff0c;让后端小伙伴分享出自己的ip地址&#xff1a; 步骤1:winr调出运行界面 步骤2&#xff1a;cmd调出命令行窗口 步骤3&#xff1a;…

JavaScript用indexOf()在字符串数组中查找子串时需要注意的一个地方

一、遇到问题 在 继续更新完善&#xff1a;C 结构体代码转MASM32代码 中&#xff0c;由于结构体成员中可能为数组类型的情况&#xff0c;因此我们在提取结构体成员信息的过程中&#xff0c;需要检测结构体成员名称字符串中是否包括 []&#xff0c;如果包括那么我们要截取[前面…

Python爬虫分布式架构 - Redis/RabbitMQ工作流程介绍

在大规模数据采集和处理任务中&#xff0c;使用分布式架构可以提高效率和可扩展性。本文将介绍Python爬虫分布式架构中常用的消息队列工具Redis和RabbitMQ的工作流程&#xff0c;帮助你理解分布式爬虫的原理和应用。 为什么需要分布式架构&#xff1f; 在数据采集任务中&#…

android2022配置opencv4android480

1&#xff0c;安装android studio2022。 2&#xff0c;下载OPENCV4ANDROID&#xff0c;解压到任意盘中。 3&#xff0c;File->New->New Project&#xff0c;选择Empty Views Activity。再选择语言&#xff0c;本文选择JAVA。 4&#xff0c;File->New->Import Modu…

麒麟系统开启root账户及自动登陆

1.首先我们通过“开始菜单t”快捷键打开命令行页面&#xff0c;然后我们通过 cd /usr/share/lightdm/lightdm.conf.d/进入对应系统目录。之后我们通过ls命令查看目录中的文件&#xff0c;找到95-ukui-greeter.conf这个文件。 2.之后我们通过命令 sudo vim 95-ukui-greeter.c…

【C++进阶(二)】STL大法--vector的深度剖析以及模拟实现

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:C从入门到精通⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学习C   &#x1f51d;&#x1f51d; vector 1. 前言2. 熟悉vector的接口函数2.1 vec…

leetcode 1022.从根到叶的二进制数之和

⭐️ 题目描述 &#x1f31f; leetcode链接&#xff1a;https://leetcode.cn/problems/sum-of-root-to-leaf-binary-numbers/description/ 代码&#xff1a; class Solution { public:int sum (TreeNode* root , int num 0) {if (root nullptr) {return 0;}int cur num r…

Docker学习笔记

Docker学习笔记 docker的作用docker的基本组成安装docker阿里云镜像加速run的流程和docker原理 docker的思想来自于集装箱。 核心思想&#xff1a; 隔离 docker可以通过隔离机制将服务器利用到极致。 虚拟机&#xff1a;在windows中装一个Vmware&#xff0c;通过这个软件可以虚…

Ubuntu本地快速搭建web小游戏网站,并使用内网穿透将其发布到公网上

文章目录 前言1. 本地环境服务搭建2. 局域网测试访问3. 内网穿透3.1 ubuntu本地安装cpolar内网穿透3.2 创建隧道3.3 测试公网访问 4. 配置固定二级子域名4.1 保留一个二级子域名4.2 配置二级子域名4.3 测试访问公网固定二级子域名 前言 网&#xff1a;我们通常说的是互联网&am…