吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 718|回复: 8
收起左侧

[学习记录] 【网络安全】Shell脚本学习

[复制链接]
st0rm 发表于 2024-11-27 11:34

声明:学习视频来自 b 站 up 主 泷羽 sec,如涉及侵权马上删除文章

声明:本文主要用作技术分享,所有内容仅供参考。任何使用或依赖于本文信息所造成的法律后果均与本人无关。请读者自行判断风险,并遵循相关法律法规。

脚本创建执行与变量使用

首先进入Linux终端【Windows系统可以使用MobaXterm中的zsh/bash终端】,可以使用 vim 创建一个 .sh 结尾的文件。
在文件的第一行写入下列三种中的其中一种,这些是脚本解释器

  • #!/bin/bash 等是一个特殊的行,称为 shebang(也称为 hashbang)。它出现在脚本文件的第一行。这一行的作用是告诉操作系统应该使用哪个解释器来执行这个脚本。

    #!/bin/bash
    #!/bin/dash
    #!/bin/sh
  • 在许多 Linux 系统中,/bin/sh(传统的 Bourne shell)实际上是指向 /bin/dash 的符号链接。dash(Debian Almquist Shell)是一个轻量级的 shell,它被设计用来提供基本的脚本执行功能,并且在启动速度和资源占用方面有一定的优势。

  • 当一个脚本没有指定解释器(即没有#!/bin/bash 或其他明确的解释器路径的 shebang 行),系统会默认使用 /bin/sh 来解释执行这个脚本。由于 /bin/sh 常常是 /bin/dash 的链接,所以在这种情况下,脚本实际上是由 dash 来执行的。

如果终端安装了 zsh 等其他脚本解释器,可以使用 #!/bin/zsh 来进行解释,或者 依然也可以使用 #!/bin/bash 等,此时执行时,依然会使用 bash 进行脚本解释,zsh 只会充当一个调用的角色。

在执行 .sh 结尾的shell 脚本时,有两种方式:

  1. chmod +x xxx.sh 添加执行权限后,使用 ./xxx.sh 执行
  2. 使用脚本解释器直接执行 sh xxx.sh / bash xxx.sh / zsh xxx.sh
  3. source xxx.sh ,该命令与使用 脚本解释器直接执行效果一致,唯一不同的地方在于该命令会将输出内容进行不同颜色标注。

变量

声明变量 name="d1"
使用变量 echo $name / echo "my name is $name"
如果使用双引号和不使用双引号的效果是一样的,都会将变量进行替换为变量值;
但如果是单引号,就不会对变量进行解析。

变量拼接

echo "my name is $name,and my age is $ageyears old"
上述例子中,变量age和字符串years之间没有空格,因为会导致解析为空,因为没有定义变量,解决办法是 使用 双引号 或 大括号 将变量引起来
echo "my name is $name,and my age is "$age"years old"
echo "my name is $name,and my age is {$age}years old"
但是使用花括号引起来的方式,在输出时,变量内容会被显示在花括号中
以上所述变量均为临时变量,变量由数字、字母和下划线组成,但不能以数字开头,也不能中间存在空格,可以使用下划线代替。

查看定义的变量

如果在终端直接定义变量,那么需要查看定义的变量时,可以使用
set | grep name
删除变量
unset name

永久环境变量和字符串显位

Windows中的环境变量在 path 下添加
which ls 在Linux下查看ls命令所在位置
echo $PATH Linux输出环境变量
查看输出结果存在/usr/bin目录什么意思呢就是当我们执行ls它会帮我们找到对应得目录做一个执行,也就是ls的完整路径应该是/usr/bin/ls脚本

根据上述描述,我们可以直接将我们的脚本添加至环境变量使其可以直接执行:

  1. 将写好的脚本 1.sh 移动到环境变量 /usr/bin 等任意一个目录下,就可以在终端直接执行 1.sh
  2. 将整个目录都添加到环境变量中,这样整个目录下的文件都可以直接执行。例如:将 /root/ 目录添加到环境变量中, /usr/bin:/bin:/usr/bin:/drives/c/windows:/root/
    1. 使用命令 export PATH=/root:$PATH
    2. export 命令是一个用于设置环境变量的关键字,使得设置的变量可以在当前shell以及由该shell启动的子进程中生效。
    3. PATH:这是一个非常重要的环境变量,它定义了系统在哪些目录下去寻找可执行程序。当你在命令行输入一个命令(如Is、cat等)时,系统会根据PATH环境变量所指定的目录顺序去查找对应的可执行文件,找到后就执行它。
    4. /root:$PATH:这里是在重新定义PATH的值。它将/root目录添加到了原有的PATH变量值的最前面(假设原有的PATH值存储在变量PATH中,这里通过:PATH的形式保留了原来的值并添加了新的部分)。这样做的结果是,当系统去查找可执行程序时,会先在/root目录下查找,然后再按照原来PATH所指定的其他目录顺序查找。

永久变量

修改用户配置文件(~/.bashrc 或 ~/.bash_profile

  • 对于 bash 用户,通常可以在用户主目录下的 ~/.bashrc 或 ~/.bash_profile 文件中添加目录到环境变量。如果使用的是 zsh,则是 ~/.zshrc。以 bash 为例,使用文本编辑器(如 vi 或 nano)打开 ~/.bashrc 文件。
    vi ~/.bashrc
  • 在文件末尾添加以下行(假设要添加 /home/user/permanent_bin 目录):
export PATH=$PATH:/home/user/permanent_bin
  • 保存并退出文件后,要使修改生效,可以在终端中运行 source ~/.bashrc。这样,以后每次打开新的终端会话,/home/user/permanent_bin 目录都会自动添加到 PATH 中。
    修改系统配置文件(全局设置,谨慎操作)
    • 在某些情况下,可能需要为系统的所有用户添加目录到环境变量。这可以通过修改系统级别的配置文件来实现,如 /etc/profile 或 /etc/environment
    • 例如,在 /etc/profile 文件中添加目录路径。使用文本编辑器打开 /etc/profile
vi /etc/profile
  • 找到合适的位置,添加类似以下的内容(假设要添加 /systemwide/bin 目录):
export PATH=$PATH:/systemwide/bin
  • 保存并退出后,这个目录就会被添加到所有用户的环境变量 PATH 中。不过,修改系统配置文件可能会影响整个系统的运行,需要谨慎操作。

字符串相关操作

假设我们想知道一个字符串的长度,比如我们想解析一个字符串的长度我们如何进行实现?
比如 name="D1" age=27 然后我们通过 echo "my name is $name,and i am $age years old"  打印完整字符串

str="hello world"
echo ${#str}
输出字符串 str 的长度

echo ${str:0:3}
输出前三个字符

脚本参数传递与数学运算

向脚本程序传递参数如何实现?

echo 执行的文件名是:$0
echo 第一个参数是:$1
echo 传递的参数作为一个字符串显示:$*
echo 传递的参数独立作为每个字符串显示:$@
echo 传递到脚本的参数个数是:$#
echo 最后命令的退出状态:$?
echo 脚本运行的当前进程ID是:$$

将上述内容添加到 2.sh 脚本中
执行脚本时, ./2.sh 1 2 这样,2.sh是第一个参数,但不是传入的第一个参数,因此 $0 就是 2.sh ,1 是传入的第一个参数,因此$1 就是 1。

由此得出 参数传递一般为 $n(n表示数字 可递增)$* 就是将参数当作统一的字符串显示出来,而 $@ 是将每个字符串当作独立的字符串显示这也是 $*$@ 的区别,$# 代表参数的个数,$? 是查看命令执行状态的,如果显示为0那么命令就是正常执行的,其他数字就是出错了,$$ 就是查看脚本当前进程id的。

在shell中进行数学运算

在shell编程中,利用expr进行运算
expr 5+10  回显 5+10
expr 5 + 10 这样写就行了5(空格)+(空格)10
以上方法适用于加法减法,如果用乘法就会报错。

expr 6\*6 (乘法是用\*  python直接使用 * 就行了,而shell要用\进行转义)

除法:
expr 6 / 3  / 是取整数
expr 6 % 3 % 是取余

混合运算
expr ( 5 + 7 ) \* 2 这样会报错,括号依然需要转义 expr \( 5 + 7 \) \* 2

通过变量进行运算
Pasted image 20241126110037.png
需要将变量内容赋值为运算表达式,使用反引号,这样在调用时,就可以直接得到运算结果。

脚本与用户交互以及 if 条件判断

使用shell脚本进行交互

利用read name age指定变量信息xiaoyu 26 ,然后利用echo $name  和 echo $age 来接收用户输入。
利用read -p "请输入您的姓名" name age
输入:xiaoyu 80 【这种方式输入时,需要用空格将两个变量分开,且一次性输入】
echo $name(接收用户的输入)
-p 用于输出后面的内容
-t 用于指定用户输入的事件,超时自动退出
-n 用于限制用户输入内容的长度

关系运算符

首先定义两个变量,然后通过if条件判断来进行两个变量直接的判断,只能对数值进行判断,无法对字符进行判断。

-eq 相等
-lt 小于
-gt 大于
-ne 不等于
else 反转
num1 = 28
num2 = 36
if [ "$num1" -eq "$num2" ] ;then
    echo 相等
else 
    echo 不相等
fi

字符串运算和逻辑运算符

字符串运算

注意方括号后的空格,以及分号后面的空格,还有变量的引号

str1="hello"
str2="hello"
if [ "$str1" = "$str2" ]; then
    echo True
else 
    echo False
fi

注意:Linux大小写敏感

str1="hello"
str2="hello"
if [ -z "$str1" ]; then
    echo True
else 
    echo False
fi

-z 参数,用于检查变量长度是否为零
-n 参数,检查变量长度是否不为零,与-t相反

str1="hello"
str2="hello"
if [ "$str1" ]; then
    echo True
else 
    echo False
fi

不设置任何参数,如果 str1为空,则返回 False,不为空,返回True。

逻辑运算

布尔运算

num1=9

if [ "$str1" != "9" ]; then
    echo num1不等于9
else 
    echo num1等于9
fi

余运算

num1=9
num2=19
if [ "$str1" != "9" -a $num2 -lt "20" ]; then
    echo True
else 
    echo False  
fi

同时满足 $num1 != 9 -a $num2 -it 20 输出true,不满足返回false
与之相反的参数是-o参数,只需要满足其中一个就可以返回true两个都不满足就返回flase

if 条件判断与 for 循环结构

if 条件判断

#!/bin/bash
# 定义变量
a=10
b=20

# 进行条件判断
if [ "$a" -eq "$b" ]; then
    echo "a=b"
elif [ "$a" -gt "$b" ]; then
    echo "a>b"
else 
    echo "没有符合上述条件"
fi

bash 脚本中,通过 “变量名=值” 的方式就可以完成变量的定义和赋值。

for 循环

for num in 1 2 3 4 5
do
    echo "The number is $num"
done

这是一段使用for循环的代码片段,常见于Shell脚本(如bash脚本)中,用于对一系列的值进行迭代操作。

循环语句 for num in 1 2 3 4 5 :
for:是循环的关键字,用于开启一个for循环结构。
num:定义了一个循环变量,在每次循环迭代过程中,这个变量会被赋予不同的值。
in 1 2 3 4 5 :指定了个值的列表,循环变量。num将会依次取这个列表中的每一个值,num会先被赋值为1,然后进行第一次循环;接着num会被赋值为2,进行第二次循环,以此类推,
直到num取完列表中的最后一个值5。

循环体
do:是与for循环起始语句配合使用的关键字,它标志着循环体的开始。在do和后面的done之间的语句就是每次循环时需要执行的操作。

输出语句 echo "The number is Snum":
echo:是一个用于在终端输出信息的命令。

循环结束
done:是for循环的结束关键字,它标志着整个for循环过程结束。

for str in "hello world"
do
    echo $str
done

上述循环中,str 只会被赋值一次,其赋值为字符串,因为值列表中只有一个字符串。

for循环与while循环

for i in `seq 1 100`
do 
    echo $i
done
for i in $(seq 1 100)
do
    echo $i
done

上述两种写法的作用都是一样的,使 i 从1取值到100,并输出。

for ((i=1; i<100; i++))
do
    echo $i
done

在for循环的双括号语法(())中:
初始化部分i=1是给循环变量i赋初值为1。
条件判断部分i<100表示只要i的值小于100,循环就会继续执行。
选代部分i++是每次循环结束后让i的值自增1,这样就能实现从1开始,每次增加1,直到1达到99(因为当1等于100时就不满足<100这个条件了),并在每次循环中通过echo输出i的当前值。

while 循环

i=1
while (( $i<=10 ))
do
    echo $i
    ((i++))
done
i=1
while (( $i<=10 ))
do
    echo $i
    let "i++"
done

代码(1)
变量初始化:
首先通过i=1这一行,将变量i初始化为1,为后续的循环操作确定起始值。

循环条件判断:
while(( $i<=10 ))是循环的条件判断部分。这里使用了双括号(())语法,它用于进行算术表达式的计算和判断。在每次循环开始前,都会检查的值是否小于等于10。只要这个条件满足,循环体内部的代码就会被执行。

循环体执行内容:
在循环体内部,首先通过 echo $i 这一行,将当前i的值输出到控制台,这样就可以看到循环过程中 i 的变化情况。
接着,通过((i++))这一语句对变量i进行自增操作。这里的(())语法同样用于算术运算,i++表示将i的值增加1。每次循环结束后,i的值就会比上一次循环时增加1,以便在下次循
环开始时进行新的条件判断。
let 命令用于 执行算术运算符,可以接收一个或多个算术表达式作为参数,并对相关变量执行算术操作。

until 循环以及函数基本创建调用

until 循环

until 直到xxx为止

i=0 # 初始化变量
until [ ! $i -lt 10 ]
do
    echo $i
    ((i++))
done

这段代码使用 until 循环结构,用于在满足特定条件之前重复执行循环体中的操作。
上述代码中,会不断输出 i 的值,并对 i 的值进行自增,直到 i 的条件不满足循环条件。
上述循环条件为 i 不小于 10 时,进行不断循环。
[] 的作用与 test 语句一致

[ ! $i -lt 10 ]
与
test! $i -lt 10

Case 语句

read -p "请您输入一个数值:" num
case $num in 
    1)echo 您输入的数字是1
    ;;
    2)echo 您输入的数字是2
    ;;
    *)echo 您输入的是其他数字
    ;;
esac

上述这段代码的功能是,从用户那里获取一个数值,根据输入的数值进行不同的处理。
如果输入是1,就输出第一个字符串。

获取用户输入:
read -p "请您输入一个数值:" num:这里使用read命令来读取用户从键盘输入的内容。p选项用于在等待输入之前向用户显示提示信息,在这个例子中,提示信息是“请您输入一个数值:,用户输入的内容会被存储到变量num中。

case $num in:这是bash,中的case语句的起始标识,用于根据变量num的不同取值来执行不同的代码块。
1)、2)和*)部分:
1):当变量num的值等于1时,会执行这部分代码。即echo您输入的数字是1,然后通过;;结束这个分支的执行。
2):同理,当num的值等于2时,执行echo您输入的数字是2,并以;;结束。
*) :这是一个通配符分支,当num的值既不等于1也不等于2时,就会执行这部分代码,即echo您输入的是其他数字,最后也以;;结束整个case语句。

基本函数学习

DemoFunc(){
    echo "hello world"
}
DemoFunc // 单独一行

上述代码是一个基本函数的创建和引用

DemoFunc(){
    echo "Hello world"
    echo "my name is $1"
}

DemoFunc xiaoyu

代码功能
这段代码定义了一个函数DemoFunc,然后调用了该函数并传递了参数xiaoyu。函数DemoFunc的功能是在被调用时,先输出固定的字符串“hello world”,接着输出一个包含参数值的字符串“My name is $1:”,这里的$1表示函数接收到的第一个参数。

代码执行过程
函数定义:
DemoFunc():这部分声明了函数名为DemoFunc(),大括号内是函数体内容。
echo “hello world":当函数被调用时,这是函数体中首先执行的语句,会将字符串“helloworld”输出到控制台。
echo “My name is $1:”:在输出“hello world”之后,会执行这条语句。由于在调用函数DemoFunc()时传递了参数xiaoyu,此时S1的值就被设置为xiaoyu,所以这条语句会输出“My
name is xiaoyu:"。

函数调用:
DemoFunc xiaoyu:这行调用了已经定义好的DemoFunc函数,并将参数xiaoyu传递给它。

多函数调用

DemoFunc(){
    echo "hello world"
    echo "My name is $1 , and my age is $2 years old"
}
DemoFunc xiaoyu 27

不同脚本的互相调用

2.sh

#!/bin/bash
echo "hello 2.sh"

3.sh

#!/bin/bash
. 2.sh
或
source 2.sh

注意:是  .空格2.sh
此时执行3.sh就会执行 2.sh的内容

变量的调用

1.sh

name="d1"
age=22

2.sh

source 1.sh
echo "my name is $1 , and my age is $2 years old"

重定向

重定向分为 输入重定向和输出重定向

ls > 1.txt
cat 1.txt

who > 1.txt
cat 1.txt

追加内容
ls >> 1.txt
cat 1.txt

如果将 ls 重定向到垃圾回收站
ls > /dev/null
不会有任何回显

输入重定向
dist.txt

/home/user/Document
/home/user/Pictures
/home/user/Music

使用 ls 命令加输入重定向,查看目录下的文件

ls -al < dist.txt

文件描述符

0:标准输入
1:标准输出
2:错误输出

例子:

ls > 3.txt 2>5.txt

如果输出正确,就会将结果输出到3.txt,如果错误,就将结果输出到5.txt

免费评分

参与人数 1吾爱币 +1 热心值 +1 收起 理由
ganbey + 1 + 1 谢谢@Thanks!

查看全部评分

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

hgxnxxgw 发表于 2024-11-27 12:00
学习留名
qqycra 发表于 2024-11-27 12:16
ljt543 发表于 2024-11-27 12:34
o276496959 发表于 2024-11-27 12:39
感谢分享
Lonelyleaves 发表于 2024-11-27 13:25
都是比较基础的了,针对目前AI的出现,感觉很多简单的脚本直接使用AI完成即可,或者针对性修改。不过还是需要掌握基础语法,才能看懂和修改
ganbey 发表于 2024-11-27 14:06
马克了,只会一些简单的bash命令,写起脚本就抓瞎。说起来shell脚本能不能帮按快捷键
tymydol 发表于 2024-11-27 14:18
都是大佬啊
iiink 发表于 2024-11-27 16:27
大赞,从基础学起来
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2025-1-5 06:22

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表