免交互 、 字符处理与高级变量

一、多行重定向

Here Document  多行重定向

定义

        使用I/O重定向的方式将命令列表提供给交互式程序

        标准输入的一种替代品

        Here Document 是标准输 入的一种替代品,可以帮助脚本开发人员不必使用临时文件来构建输入信息,而是直接就地 生产出一个<font color='red'>文件</font><font color='cornflowerblue'>并用作</font><font color='red'>命令</font>的标准输入,Here Document 可以与非交互式程序和命令一起使用

语法格式:

命令  << 标记

....

......

标记

注意事项:

  1. 标记可以使用任意的合法字符(通用的字符是EOF)

  2. 结尾的标记一定要顶格写,前面不能有任何字符(包括空格)

  3. 结尾的标记后面也不能有任何字符(包括空格)

  4. 开头标记前后空格会被省略掉

  5. 单引号 变量双引号 —

例一:

        使用 wc -l 命令后面直接跟文件名就可以统计文件内有多少行内容,将要统计的内容置于标记“EOF” 之间,直接将内容传给 wc -l 来统计。

[root@localhost opt]#vim 1.sh
#!/bin/bash
wc -l <<EOF
line 1
line 2
EOF

[root@localhost opt]#bash 1.sh 
2


1. 此处含义解释: wc -l   <font color='purple'>统计行数</font>

2. <<EOF

   .......

   EOF        

   之间的内容 传入脚本做为参数给 wc-l
例二:

通常使用 read 命令接收用户的输入值时会有交互过程

[root@localhost ~]# read -p "请输入一个数字" ack
请输入一个数字8
[root@localhost ~]# echo $ack 
8



[root@localhost opt]#read a <<EOF
> 1122
> EOF
[root@localhost opt]#echo $a
1122
例三:

        使用 passwd命令设置密码

        EOF 标记之间的两行是输入的密码和确认密码,两行内容必须保持一致,否则密 码设置不成功。此脚本执行后不会输出任何信息,可另开一个终端使用 jerry 用户登录,输 入新修改的密码来验证密码是否修改正确

[root@localhost opt]#passwd zhangsan <<EOF
> 1234
> 1234
> EOF

注意密码的格式要求

例四:

编写yum仓库的内容

[root@localhost etc]#cat <<EOF >/etc/yum.repos.d/local.repo
> [local]
> name=local
> baseurl=file:///mnt
> gpgcheck=0
> EOF

[root@localhost etc]#cat /etc/yum.repos.d/local.repo 
[local]
name=local
baseurl=file:///mnt
gpgcheck=0

Here Document 变量设定

Here Document 也支持使用变量,如果标记之间有变量被使用,会先替换变量值。如 果想要将一些内容写入文件,除了常规的方法外,也可以使用 Here Document。如果写入 的内容中包含变量,在写入文件时要先将变量替换成实际值,在结合 cat 命令完成写入。

例一:
[root@localhost ~]# cat <<EOF
> hello
> world
> !!!
> EOF
hello
world
!!!


[root@localhost data]#cat >>/data/1 <<EOF
> 123
> asd
> zxc
> EOF
#把输入内容追加到/data/1文件中。
[root@localhost data]#cat 1
123
asd
zxc



#####可以使用变量

#!/bin/bash
file="gt.txt"
qa="school"

cat >$file <<EOF
go to $qa
EOF
#这些意思是把输入的内容导入到gt.txt文件里。
cat $file



[root@localhost data]#bash g.sh 
go to school

[root@localhost data]#cat gt.txt 
go to school
例二:

在引用的变量上,加上 " " 完美引用

[root@localhost ~]# vim eof.sh
#!/bin/bash
var="today i will good study"
myvar=$(cat <<EOF
this is my day
$var
this is good day
EOF
)

echo $myvar

[root@localhost ~]#bash eof.sh
this is my day today i will good study this is good day

给脚本里 echo $myvar 变量加上双引号 echo "$myvar"  完美复刻所有格式。

[root@localhost ~]# vim eof.sh
#!/bin/bash
var="today i will good study"
myvar=$(cat <<EOF
this is my day
$var
this is good day
EOF
)

echo "$myvar"


[root@localhost data]#bash 22.sh 
this is my day
today i will good study
this is good day
例三:

在标记符上加上  " "  表示不引用变量显示原来的意思

[root@localhost ~]# vim eof.sh
#!/bin/bash
var="today i will good study"
myvar=$(cat <<"EOF"            #这里EOF加了双引号
this is my day
$var
this is good day
EOF
)

echo "$myvar"



[root@localhost ~]# ./eof.sh 
this is my day
$var
this is good day
#可以看到变量未被引用,表示原来意思。
例四:

在标记符前加上 - 不显示tab键但是显示空格键

[root@localhost ~]# vim eof.sh

#!/bin/bash
var="today i will good study"
myvar=$(cat <<-EOF
        this is my day   #这里用的TAB
$var
   this is good day    #这里用的空格
EOF
)

echo "$myvar"


[root@localhost data]#bash eof.sh 
this is my day
today i will good study
   this is good day
#可以看到TAB了的行不显示。

例五:

前面加上冒号表示注册
:<<EOF


EOF

二、 expect 免交互

定义:

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

你需要检查是否有 expect 和 tcl 软件包

rpm -q expect

rpm -q tcl

yum install -y expect     #yum安装expect

格式:

expect     [选项]     [ -c cmds ]     [ [ -[f|b] ] cmdfile ]     [ args ]

expect中相关命令

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

spawn  passwd root

  • expect 从进程接收字符串
  • send 用于向进程发送字符串
  • exp_continue 匹配多个字符串在执行动作后加此命令
  • interact 允许用户交互
  • expect  eof   结束符

例子:

[root@localhost data]#expect
expect1.1> expect "hi" {send "hello\n"}
hihisjs
hello
expect1.2> 
#只要屏幕上出现hi ,输出hello并换行。

要先安装expect

#####免交互,传输文件#####
[root@localhost data]#vim scp.exp
#!/usr/bin/expect
spawn scp /etc/fstab 192.168.91.101:/mnt
expect {
  "yes/no" { send "yes\n";exp_continue }
  "password" { send "123123\n" }
}

expect eof

[root@localhost data]#expect scp.exp    #注意使用expect运行这个脚本
spawn scp /etc/passwd 192.168.80.10:/mnt
The authenticity of host '192.168.80.10 (192.168.80.10)' can't be established.
ECDSA key fingerprint is SHA256:x3FGetAdEbA5tIBXXkLnxfIj3rVQz34pGBRZ1MH/X38.
ECDSA key fingerprint is MD5:cb:3f:b4:60:6c:b3:74:19:56:76:80:ea:7c:00:13:be.
Are you sure you want to continue connecting (yes/no)? yes
Warning: Permanently added '192.168.80.10' (ECDSA) to the list of known hosts.
root@192.168.80.10's password: 
passwd                                         100% 2095     4.1MB/s   00:00    

基本命令介绍:

(1)脚本解释器

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

#!/usr/bin/expect

(2)spawn

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

例: spawn passwd root

(3)expect

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

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

(4)send

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

expect 书写方式:

方式一:
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"}
}

#这里 \r \n 都是回车

(5)  结束符

expect    eof

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

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

interact

        执行完成后保持交互状态, 把控制权交给控制台,会停留在目标终端而不是退回到原终端,这时候就可以手工操作了,interact后命令不再起作用,比如interact后添加exit,并不会退出root用户。而如果没有interact则登录完成后会退出,而不是留在远程终端上。

        使用interact会保持在终端而不会退回原终端,比如切换到root用户,会一直在root用户状态下;比如ssh到另一台服务器,会一直在目标服务器终端而不会切回原服务器。

需要注意的是,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命令去执行脚本

远程拷贝

[root@localhost data]#vim kaobei.exp

#!/usr/bin/expect
spawn scp /etc/passwd 192.168.80.10:/data
expect {
  "yes/no" {send "yes\n";exp_continue}
  "password"  {send "123\n"}
}
expect eof

[root@localhost data]#expect kaobei.exp

ssh远程连接

[root@localhost data]#vim ssh.exp

#!/usr/bin/expect
spawn ssh 192.168.80.10
expect {
   "yes/no"  {send "yes\n";exp_continue}
   "password" {send "123\n"}

}
interact
#执行完成后保持交互状态

[root@localhost data]#expect ssh.exp

变量

[root@localhost data]#vim ssh.exp

#!/usr/bin/expect
set user root
set ip 192.168.80.10
set pw 123

set timeout 10
#设置超时时间

spawn ssh $user@$ip
expect {
   "yes/no"  {send "yes\n";exp_continue}
   "password" {send "$pw\n"}

}
interact
#执行完成后保持交互状态

[root@localhost data]#expect ssh.exp

接收参数(位置变量)

远程ssh脚本

[root@localhost data]#vim ssh2.exp

#!/usr/bin/expect
set timeout 10

set hostname [lindex $argv 0]
#相当于hostname=$1
set ip [lindex $argv 1]
#相当于ip=$2
set password [lindex $argv 2]
#相当于password=$3

spawn ssh $hostname@$ip
expect {
  "Connection refused" exit
  "No route to host"  exit
  "yes/no" {send "yes\n";exp_continue}
  "password" {send "$password\n"}
}
interact

[root@localhost data]#expect ssh2.exp root 192.168.80.10 123
                                     用户名   IP地址      密码

切换用户脚本

[root@localhost data]#su - w
上一次登录:四 2月  1 14:28:59 CST 2024pts/1 上
[w@localhost ~]$ 
[w@localhost ~]$ vim su.exp
[w@localhost ~]$ expect su.exp root 123
spawn su root
密码:
[root@localhost w]#hello root!
#这里 当前目录 出现 w 是因为没有完全切换用户。

#!/usr/bin/expect
set timeout 5
#设置超时时间

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

spawn su $name
#可以写成 su - $name 完全切换用户。

expect {
  "密码:" {send "$pd\n"}
}

expect {
  "*]#"  {send_user "hello root!"}
  #这里 * 的意思是字符出现0到无数次且以]#结尾,只要屏幕上出现这种情况就回显hello root!(跟echo相同)
}
interact

使用bash 编写 expect脚本

[root@localhost data]#vim ssh3.sh

#!/bin/bash
name=$1
ip=$2
pw=$3

/usr/bin/expect <<EOF
spawn ssh $name@$ip

expect {
  "yes/no" {send "yes\n";exp_continue}
  "password" {send "$pw\n"}
}
expect "*]#"
send "ifconfig ens33\rexit\r"
expect eof
EOF

执行结果:

[root@localhost data]#bash ssh3.sh root 192.168.80.10 123
spawn ssh root@192.168.80.10
root@192.168.80.10's password: 
Last login: Thu Feb  1 15:30:54 2024 from 192.168.80.7
[root@localhost ~]#ifconfig ens33
ens33: flags=4163<UP,BROADCAST,RUNNING,MULTICAST>  mtu 1500
        inet 192.168.80.10  netmask 255.255.255.0  broadcast 192.168.80.255
        inet6 fe80::8a14:98:a26f:9ab6  prefixlen 64  scopeid 0x20<link>
        ether 00:0c:29:70:48:93  txqueuelen 1000  (Ethernet)
        RX packets 114990  bytes 7674276 (7.3 MiB)
        RX errors 0  dropped 0  overruns 0  frame 0
        TX packets 20470  bytes 6147335 (5.8 MiB)
        TX errors 0  dropped 0 overruns 0  carrier 0  collisions 0

[root@localhost ~]#exit
登出
Connection to 192.168.80.10 closed.

如何对两台主机进行操作?

#!/bin/bash
net=192.168.80.
pw=123
iplist="
10
11
"

for i in $iplist
do
{
  ip=$net$i

/usr/bin/expect <<EOF

spawn ssh root@$ip

expect {
  "yes/no" {send "yes\n";exp_continue}
  "password" {send "$pw\n"}
}

expect {
  "*]#" {send "useradd pamu\n"}
  "*]#" {send "echo 123|passwd pamu --stdin\n"}
  "*]#" {set timeout 1;send "exit\n"}
#不能加exp_continue,不然会一直执行。
}
expect eof
EOF
} &
done

三、字符处理

字符串切片

3.1 基于偏移变量取字符

一、取字符串长度

${#var}
#返回字符串变量var的字符的长度,一个汉字算一个字符

例子:

[root@localhost ~]#str=abc123你好
#定义变量
[root@localhost ~]#echo ${#str}
#使用 字符公式取长度
8

[root@localhost ~]#echo {a..z}
a b c d e f g h i j k l m n o p q r s t u v w x y z
[root@localhost ~]#echo {a..z}|tr -d " "
abcdefghijklmnopqrstuvwxyz
[root@localhost ~]#str=`echo {a..z}|tr -d " "`
[root@localhost ~]#echo $str
abcdefghijklmnopqrstuvwxyz
[root@localhost ~]#echo ${#str}
26


数组:
[root@localhost ~]#a=(a b c v)
[root@localhost ~]#echo ${#a[*]}
4

跳过左边的字符

${var:offset}
#返回字符串变量var中从第offset个字符后(不包括第offset个字符)的字符开始,到最后的部分,offset的取值在0 到 ${#var}-1 之间(bash4.2后,允许为负值)
${var:offset:number}
#返回字符串变量var中从第offset个字符后(不包括第offset个字符)的字符开始,长度为number的部分

例子:

[root@localhost ~]#str=`echo {a..z}|tr -d " "`
[root@localhost ~]#echo $str
abcdefghijklmnopqrstuvwxyz


[root@localhost ~]#echo ${str:2}
#跳过最左边的两个字符  跳过ab
cdefghijklmnopqrstuvwxyz


[root@localhost ~]#echo ${str:2:3}
#跳过前两个, 往后取三个
cde


[root@localhost ~]#echo ${str:0:2}
#取前两个字符
ab

取字符串右边的字符

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

例子:

[root@localhost ~]#echo ${str}
abcdefghijklmnopqrstuvwxyz
[root@localhost ~]#echo ${str:-3}
#不加  空格没有任何变化
abcdefghijklmnopqrstuvwxyz

[root@localhost ~]#echo ${str: -3}
#取字符串右边的  三个字符
xyz

掐头去尾

${var:offset:-length}
#从最左侧跳过offset字符,一直向右取到距离最右侧lengh个字符之前的内容,即:掐头去尾

例子:

[root@localhost ~]#echo ${str:2: -3}
cdefghijklmnopqrstuvw
#跳过前两个字符且跳过倒数三个字符

取倒数的范围

${var: -length:-offset}
#先从最右侧向左取到length个字符开始,再向右取到距离最右侧offset个字符之间的内容,注意:-length前空格,并且length必须大于offset

例子:

[root@localhost ~]#echo ${str: -6}
uvwxyz
[root@localhost ~]#echo ${str: -6: -1}
uvwxy
#倒数6个 到 倒数 1个。



[root@localhost ~]#echo ${str: -3:1}
x
#取字符串右边三个字符 后 取一个字符

3.2基于模式取子串

删左留右

${var#*word}
#其中word可以是指定的任意字符,自左而右,查找var变量所存储的字符串中,第一次出现的word, 删除字符串开头至第一次出现word字符串(含)之间的所有字符,即懒惰模式,以第一个word为界删左留右

#同上,贪婪模式,不同的是,删除的是字符串开头至最后一次由word指定的字符之间的所有内容,即贪婪模
式,以最后一个word为界删左留右
${var##*word}

例子:

懒惰模式:
[root@localhost ~]#url="http://www.baidu.com/index.html"
[root@localhost ~]#echo ${url#*/}
#从左边开始匹配,遇到/  就把/左边的都删除包括 /  本身
/www.baidu.com/index.html


[root@localhost ~]#url="http://www,baidu.com:8080/index.html"
[root@localhost ~]#echo ${url#*:}
//www,baidu.com:8080/index.html



贪婪模式:
[root@localhost data]#url="http://www.baidu.com/index.html"
[root@localhost data]#
[root@localhost data]#echo ${url##*/}
index.html
#从左边开始匹配 到最后一个 / 并把左边的都删除包括 /  本身

[root@localhost data]#echo ${url##*.}
html
#从左边开始匹配 到最后一个 . 并把左边的都删除包括 .  本身

删右留左

${var%word*}
#其中word可以是指定的任意字符,功能:自右而左,查找var变量所存储的字符串中,第一次出现的word, 删除字符串最后一个字符向左至第一次出现word字符串(含)之间的所有字符,即懒惰模式,以从右向左的第一个word为界删右留左


${var%%word*}
#同上,只不过删除字符串最右侧的字符向左至最后一次出现word字符之间的所有字符,即贪婪模式,以从右向左的最后一个word为界删右留左

例子:

懒惰模式:

[root@localhost ~]#echo $url
http://www.baidu.com:8080/index.html
[root@localhost ~]#echo ${url%:*}
#懒惰模式, 遇到  :  会把右边删除留下左边
http://www.baidu.com



贪婪模式:

[root@localhost ~]#echo ${url%%:*}
#贪婪模式, 会一直找到所有的  :  会把左边删除留下右边
http

3.3 查找替换

${var/pattern/substr}
${变量/搜索的字符串/修改的字符串}
#查找var所表示的字符串中,第一次被pattern所匹配到的字符串,以substr替换之

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

${var/#pattern/substr}
#查找var所表示的字符串中,行首被pattern所匹配到的字符串,以substr替换之
#  在 此处  #代表开头


${var/%pattern/substr}
#查找var所表示的字符串中,行尾被pattern所匹配到的字符串,以substr替换之
# 在此处  % 代表结尾

例子:

[root@localhost data]#url=http://www,baidu.com:8080/index.html

[root@localhost data]#echo ${url/http/https}
https://www,baidu.com:8080/index.html
#把 http换成  https

[root@localhost data]#echo ${url}
http://www,baidu.com:8080/index.html
#只是改变显示,内容没变。


[root@localhost ~]#echo ${url/:/+}
http+//www.baidu.com:8080/index.html
[root@localhost ~]#echo ${url//:/+}
#把所有的冒号换成加号
http+//www.baidu.com+8080/index.html




[root@localhost ~]#echo ${user/#lisi/ls}
ls:x:1003:1003::/home/lisi:/bin/bash
#开头的lisi换成ls

[root@localhost ~]#echo ${user/%bash/nologin}
lisi:x:1003:1003::/home/lisi:/bin/nologin
#结尾的bash 换成nologin

3.4 查找并删除

${var/pattern}
#删除var表示的字符串中第一次被pattern匹配到的字符串

${var//pattern}
删除var表示的字符串中所有被pattern匹配到的字符串

${var/#pattern}
删除var表示的字符串中所有以pattern为行首匹配到的字符串

${var/%pattern}
删除var所表示的字符串中所有以pattern为行尾所匹配到的字符串

例子:

[root@localhost data]#useradd lisi
[root@localhost ~]#user=`getent passwd lisi`


[root@localhost ~]#echo ${user}
lisi:x:1003:1003::/home/lisi:/bin/bash

[root@localhost ~]#echo ${user/lisi/}
#没有换的就是  删除
:x:1003:1003::/home/lisi:/bin/bash

3.5 大小写转换
 

${var^^}
#把var中的所有小写字母转换为大写


${var,,}
#把var中的所有大写字母转换为小写

例子:

[root@localhost data]#zxc=ABCDasdf
[root@localhost data]#echo ${zxc^^}
ABCDASDF
[root@localhost data]#echo ${zxc,,}
abcdasdf
[root@localhost data]#echo ${zxc}
ABCDasdf

四、高级变量

4.1 高级变量赋值

变量配置方式str没有配置str为空字符串str已配置非为空字符串
var=${str-expr}var=exprvar=var=$str
var=${str:-expr}var=exprvar=exprvar=$str
var=${str+expr}var=var=exprvar=expr
var=${str:+expr}var=var=var=expr
var=${str=expr}

str=expr

var=expr

str=不变

var=

str 不变

var=$str

var=${str:=expr}

str=expr

var=expr

str=expr

var=expr

str 不变

var=$str

var=${str?expr}expr 输出至 stderrvar=var=$str
var=${str:?expr}expr 输出至 stderrexpr 输出至 stderrvar=$str

例子:

[root@localhost ~]#unset str   #使str为没有配置状态
[root@localhost ~]#var=${str-expr}
[root@localhost ~]#echo $var
expr
#var=expr


[root@localhost ~]#str=" "   #使str为空字符串
[root@localhost ~]#var=${str-expr}
[root@localhost ~]#echo $var

[root@localhost ~]#
#var=



[root@localhost ~]#str=asa      #使str为已配置状态(不是空字符串)
[root@localhost ~]#var=${str-expr}
[root@localhost ~]#echo $var
asa
#var=$str

4.2 变量的间接引用

        eval命令

        eval命令将会首先扫描命令行进行所有的置换,然后再执行该命令。该命令适用于那些一次扫描无法实现其功能的变量,该命令对变量进行两次扫描

        例子:

[root@localhost ~]#n=10
[root@localhost ~]#echo {1..$n}
{1..10}

#这是因为 在输出时只扫描了一次$n,{1..10}未被扫描,所以输出的时{1..10}


[root@localhost ~]#eval echo {1..$n}
1 2 3 4 5 6 7 8 9 10



[root@localhost ~]#i=a
[root@localhost ~]#j=1
[root@localhost ~]#$i$j=hello
bash: a1=hello: 未找到命令...

[root@localhost ~]#eval $i$j=hello   #相当于a1=hello
[root@localhost ~]#echo $a1
hello
  • 26
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值