字符串截取及切割

1案例1:字符串截取及切割

1.1问题

使用shell完成各种Linux运维任务时,一旦涉及到判断、条件测试等相关操作时,往往需要对相关的命令输出进行过滤,提取出符合要求的字符串。

本案例要求熟悉字符串的常见处理操作,完成以下任务练习:

  • 参考PPT示范操作,完成字符串截取、替换等操作
  • 根据课上的批量改名脚本,编写改进版renfilex.sh:能够批量修改当前目录下所有文件的扩展名,修改前/后的扩展名通过位置参数$1、$2提供

1.2方案

字符串截取的三种方法:

  • ${变量名:起始位置:长度}
  • expr substr "$变量名" 起始位置 长度
  • echo $变量名|cut -b 起始位置-结束位置

字符串替换的两种方法:

  • 只替换第一个匹配结果:${变量名/old/new}
  • 替换全部匹配结果:${变量名//old/new}

字符串掐头去尾:

  • 从左向右,最短匹配删除:${变量名#*关键词}
  • 从左向右,最长匹配删除:${变量名##*关键词}
  • 从右向左,最短匹配删除:${变量名%关键词*}
  • 从右向左,最长匹配删除:${变量名%%关键词*}

1.3步骤

实现此案例需要按照如下步骤进行。

步骤一:字符串的截取

1)方法一,使用${}表达式

格式:${变量名:起始位置:长度}

使用${}方式截取字符串时,起始位置是从0开始的。

定义一个变量phone,并确认其字符串的长度:

[root@svr5 ~]# phone="13788768897"
[root@svr5 ~]# echo ${#phone}
11                                         //包括11个字符

使用${}截取时,起始位置可以省略,省略时从第一个字符开始截。比如,以下操作都可以从左侧开始截取前6个字符:

[root@svr5 ~]# echo ${phone:0:6}
137887

或者

[root@svr5 ~]# echo ${phone::6}
137887

因此,如果从起始位置1开始截取6个字符,那就变成这个样子了:

[root@svr5 ~]# echo ${phone:1:6}
378876

2)方法二,使用expr substr

格式:expr substr "$变量名" 起始位置 长度

还以前面的phone变量为例,确认原始值:

[root@svr5 ~]# echo $phone
13788768897

使用expr substr截取字符串时,起始编号从1开始,这个要与${}相区分。

从左侧截取phone变量的前六个字符

[root@svr5 ~]# expr substr "$phone" 1 6
137887

从左侧截取phone变量,从第9个字符开始,截取3个字符:

[root@svr5 ~]# expr substr "$phone" 9 3
897

 3)方式三,使用cut分割工具

格式:echo $变量名|cut -b 起始位置-结束位置

选项-b表示按字截取字符,其中起始位置、结束位置都可以省略。当省略起始位置时,视为从第一个字符开始(编号也是从1开始,与expr类似),当省略结束位置时,视为截取到最后。

[root@svr5 ~]# echo $phone
13788768897

从左截取前6个字符,可执行以下操作:

[root@svr5 ~]# echo $phone | cut -b 1-6
137887

 从第8个字符截取到末尾:

[root@svr5 ~]# echo $phone | cut -b 8-
8897

 只截取单个字符,比如第9个字符:

[root@svr5 ~]# echo $phone | cut -b 9
8

 截取不连续的字符,比如第3、5、8个字符:

[root@svr5 ~]# echo $phone | cut -b 3,5,8
788

4)一个随机密码的案例

版本1:

[root@svr5 ~]# vim rand.sh
#!/bin/bash
x=abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789
//所有密码的可能性是26+26+10=62(0-61是62个数字)
num=$[RANDOM%62]
pass=${x:num:1}

版本2:

[root@svr5 ~]# vim rand.sh
#!/bin/bash
x=abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789
//所有密码的可能性是26+26+10=62(0-61是62个数字)
pass=''
for i in {1..8}
do
num=$[RANDOM%62]
tmp=${x:num:1}
pass=${pass}$tmp
done
echo $pass

步骤二:字符串的替换

1)只替换第1个字符串

格式:${变量名/old/new}

还以前面的phone变量为例,确认原始值:

[root@svr5 ~]# echo $phone
13788768897

将字符串中的1个8替换为X:

[root@svr5 ~]# echo  ${phone/8/X}
137X8768897

3)替换全部字符串

格式:${变量名//old/new}

将phone字符串中的所有8都替换为X:

[root@svr5 ~]# echo  ${phone//8/X}
137XX76XX97

步骤三:字符串的匹配删除

以处理系统默认的账户信息为例,定义变量A:

[root@svr5 ~]# A=`head -1 /etc/passwd`
[root@svr5 ~]# echo $A
root:x:0:0:root:/root:/bin/bash

1)从左向右,最短匹配删除

格式:${变量名#*关键词}

删除从左侧第1个字符到最近的关键词":"的部分,*做通配符理解:

[root@svr5 ~]# echo ${A#*:}
x:0:0:root:/root:/bin/bash

2)从左向右,最长匹配删除

格式:${变量名##*关键词}

删除从左侧第1个字符到最远的关键词":"的部分:

[root@svr5 ~]# echo $A                      //确认变量A的值
root:x:0:0:root:/root:/bin/bash
[root@svr5 ~]# echo ${A##*:}
/bin/bash

3)从右向左,最短匹配删除

格式:${变量名%关键词*}

删除从右侧最后1个字符到往左最近的关键词":"的部分,*做通配符理解:

[root@svr5 ~]# echo ${A%:*}
root:x:0:0:root:/root

4)从右向左,最长匹配删除

格式:${变量名%%关键词*}

删除从右侧最后1个字符到往左最远的关键词":"的部分:

[root@svr5 ~]# echo ${A%%:*}
root

步骤四:编写renfilex.sh脚本

创建一个测试文件

[root@svr5 ~]# mkdir rendir
[root@svr5 ~]# cd rendir
[root@svr5 rendir]# touch {a,b,c,d,e,f,g,h,i}.doc
[root@svr5 rendir]# ls
a.doc  b.doc  c.doc  d.doc  e.doc  f.doc  g.doc  h.doc  i.doc

1)批量修改文件扩展名的脚本

脚本用途为:批量修改当前目录下的文件扩展名,将.doc改为.txt

脚本内容参考如下:

[root@svr5 rendir]# vim renfile.sh
#!/bin/bash
for i in `ls *.doc`            #注意这里有反引号
do
    mv $i  ${i%.*}.txt
done
[root@svr5 ~]# chmod +x renfile.sh

测试脚本:

[root@svr5 rendir]# ./renfile.sh
[root@svr5 rendir]# ls
a.txt  b.txt  c.txt  d.txt  e.txt  f.txt  g.txt  h.txt  i.txt

2)改进版脚本(批量修改扩展名)

通过位置变量$1、$2提供更灵活的脚本,改进的脚本编写参考如下:

[root@svr5 rendir]# vim ./renfile.sh
#!/bin/bash
#version:2
for i in `ls *.$1`
do
    mv $i  ${i%.*}.$2
done

3)验证、测试改进后的脚本

将*.doc文件的扩展名改为.txt:

[root@svr5 rendir]# ./renfile.sh txt doc

将*.doc文件的扩展名改为.mp4:

[root@svr5 rendir]# ./renfile.sh doc mp4


 

发布了117 篇原创文章 · 获赞 2 · 访问量 3380

猜你喜欢

转载自blog.csdn.net/weixin_41176080/article/details/103783597