ls
, cd
, mkdir
, rm
等)。Shell脚本简介
基础Shell命令
ls
:列出目录内容。cd [目录]
:更改当前目录。mkdir [目录名]
:创建新目录。rm [文件或目录]
:删除文件或目录。echo [文本]
:显示一行文本。编写和执行简单脚本
my_script.sh
。echo "Hello, Shell!"
。bash my_script.sh
来执行脚本。test_dir
的目录,然后在该目录中创建一个名为hello.txt
的文件,并在文件中写入Hello, Shell!
。create_dir_and_file.sh
。#!/bin/bash
mkdir test_dir
echo "Hello, Shell!" > test_dir/hello.txt
3. 保存文件,并在终端中运行脚本。
4. 检查test_dir
目录和hello.txt
文件是否已创建,并查看文件内容。
深入基础命令
cp [源文件] [目标位置]
:复制文件或目录。mv [源文件] [目标位置]
:移动文件或目录,也可用于重命名。grep [模式] [文件]
:搜索文件中的文本。find [路径] [选项]
:在目录中查找文件。命令参数和选项
ls -l
(以长格式列出文件详细信息),grep -i
(忽略大小写进行搜索)。管道和重定向
|
:将一个命令的输出作为另一个命令的输入。例如,ls | grep "txt"
会找出所有包含"txt"的文件名。>
和 >>
:用于将命令的输出保存到文件中。例如,echo "Hello" > file.txt
会创建或覆盖文件file.txt
,并写入"Hello"。practice
的新目录。practice
目录中创建几个测试文件。grep
命令查找包含特定文本的文件。grep
命令的输出重定向到一个新文件中。#!/bin/bash
mkdir practice
cd practice
echo "Test file 1" > test1.txt
echo "Another test file 2" > test2.txt
echo "Different file 3" > test3.txt
grep "test" *.txt > results.txt
控制结构
if-else
):根据条件执行不同的命令。if [ 条件 ]; then
命令
else
另一些命令
fi
循环(for
,while
):重复执行一系列命令。
for
循环示例:
for i in 1 2 3; do
echo $i
done
while
循环示例:
count=1
while [ $count -le 3 ]; do
echo $count
((count++))
done
脚本中的比较符号:
shell脚本中-eq、-ne、-gt、-ge、-lt、-le_shell脚本中-gt-CSDN博客
文件和字符串操作
cat
或循环逐行读取。if [ "$str1" == "$str2" ]; then
。#!/bin/bash
echo -e "2\n4\n6" > numbers.txt
total=0
while read num; do
total=$((total + num))
done < numbers.txt
echo "Total sum is: $total"
#!/bin/bash
- 这是一个"shebang"行,用于指示脚本应该用哪个解释器来执行。在这个例子中,它指定使用
bash
来执行这个脚本。
echo -e "2\n4\n6" > numbers.txt
echo
命令用于输出文本。-e
选项允许echo
解释转义字符,如\n
(新行)。"2\n4\n6"
是被输出的字符串,包含数字2、4和6,每个数字后面跟着一个换行符。>
是重定向符号,用于将echo
命令的输出写入numbers.txt
文件。如果文件已存在,则会被覆盖。
total=0
- 这行代码初始化变量
total
为0。在循环中,total
将用于累加所有数字。
while read num; do
- 这是一个
while
循环的开始。循环会一直执行,直到读取文件的行结束。read num
命令读取输入(在这个例子中是文件numbers.txt
的每一行)并将其存储在变量num
中。
total=$((total + num))
- 这行代码更新变量
total
的值。它使用$((...))
来进行算术运算,将total
的当前值加上变量num
的值。
done < numbers.txt
- 这是
while
循环的结束部分。< numbers.txt
将numbers.txt
文件的内容重定向到while
循环中,即read
命令读取的输入来自于这个文件。
echo "Total sum is: $total"
- 最后,这行代码输出计算得到的总和。
- 这里使用了双引号来允许变量
$total
在字符串中被扩展(即替换为其值)。
Shell函数
function_name () {
# 函数体
命令
}
文本处理工具
sed
:流编辑器,用于执行文本替换、删除、插入等操作。awk
:强大的文本分析工具,适用于模式扫描和处理。awk
来分析文件并输出特定列。sed
或awk
对文本文件进行处理。#!/bin/bash
# 定义一个函数
print_file () {
cat $1
}
# 使用sed来替换文本
replace_text () {
sed -i 's/old/new/g' $1
}
echo "Creating a sample file..."
echo "This is old text" > sample.txt
echo "Before replacement:"
print_file sample.txt
echo "Replacing text..."
replace_text sample.txt
echo "After replacement:"
print_file sample.txt
#!/bin/bash
- 这行是shebang,它告诉系统使用
bash
解释器来执行这个脚本。定义函数
print_file
:
print_file () {
- 这行开始定义一个名为
print_file
的函数。cat $1
cat
命令用于显示文件的内容。$1
是函数的第一个参数,在这里它将被解释为传递给函数的文件名。}
- 这行结束函数定义。
定义函数
replace_text
:
replace_text () {
- 开始定义一个名为
replace_text
的函数。sed -i 's/old/new/g' $1
- 这行使用
sed
命令对文件进行文本替换。-i
表示原地编辑文件。's/old/new/g'
是sed
的替换命令,意味着将所有出现的“old”替换为“new”。$1
代表函数的第一个参数,即要处理的文件名。}
- 结束函数定义。
脚本主体:
echo "Creating a sample file..."
- 输出提示信息。
echo "This is old text" > sample.txt
- 创建一个名为
sample.txt
的新文件,并写入文本“This is old text”。echo "Before replacement:"
- 输出提示信息。
print_file sample.txt
- 调用
print_file
函数,将sample.txt
作为参数传递,打印文件内容。echo "Replacing text..."
- 输出提示信息。
replace_text sample.txt
- 调用
replace_text
函数,将sample.txt
作为参数传递,对文件进行文本替换。echo "After replacement:"
- 输出提示信息。
print_file sample.txt
- 再次调用
print_file
函数,打印替换文本后的文件内容。
awk
和sed
进行高级文本处理。高级文本处理
awk
高级用法:处理复杂文本格式,如字段提取、模式匹配、统计信息等。sed
高级用法:进行更复杂的文本替换和删除操作。脚本调试
set -x
和set +x
在脚本中开启和关闭调试模式。bash -x your_script.sh
来运行脚本,以显示每个命令及其参数。awk
处理文本文件,并包含脚本调试元素。#!/bin/bash
set -x # 开启调试
# 使用awk处理文本
process_text_with_awk () {
awk '{print $1, $3}' $1
}
echo "Creating a sample file..."
echo -e "1 apple 3.5\n2 banana 4.2\n3 cherry 2.8" > fruits.txt
echo "Processed text:"
process_text_with_awk fruits.txt
set +x # 关闭调试
set -x
- 这行命令开启了脚本的调试模式。在调试模式下,脚本会打印每个命令及其扩展后的参数到标准输出,帮助理解脚本的执行过程。
定义函数
process_text_with_awk
:
process_text_with_awk () {
- 这行开始定义一个名为
process_text_with_awk
的函数。awk '{print $1, $3}' $1
- 使用
awk
命令处理文本。awk
会读取传入的文件(在这个例子中是函数的第一个参数$1
),并对每一行执行花括号内的操作。这里的操作是打印每行的第一个和第三个字段,字段默认由空格分隔。}
- 这行结束函数的定义。
创建示例文件:
echo "Creating a sample file..."
- 打印提示信息。
echo -e "1 apple 3.5\n2 banana 4.2\n3 cherry 2.8" > fruits.txt
- 创建一个名为
fruits.txt
的新文件,并写入三行文本。每行包含一个数字、一个水果名和一个价格,字段间由空格分隔。调用函数处理文本:
echo "Processed text:"
- 打印提示信息。
process_text_with_awk fruits.txt
- 调用
process_text_with_awk
函数,并传入fruits.txt
作为参数。
set +x
- 这行命令关闭了脚本的调试模式。执行到这一行时,脚本会停止打印后续命令的执行细节。
错误处理
set -e
使脚本在出现错误时立即退出。trap
命令捕获并处理信号和脚本退出。性能优化
#!/bin/bash
set -e # 出现任何错误时退出脚本
trap 'echo "Error occurred at $LINENO"; exit 1' ERR
# 一个可能失败的操作
possibly_failing_operation () {
false # 这个命令会失败
}
# 性能优化的操作
optimized_operation () {
for i in {1..1000}; do
: # 内建的空操作
done
}
echo "Running possibly failing operation..."
possibly_failing_operation
echo "Running optimized operation..."
optimized_operation
echo "Script completed successfully."
trap 'echo "Error occurred at $LINENO"; exit 1' ERR
trap
命令用于在脚本接收到指定的信号时执行特定的操作。在这里,它被配置为在脚本因任何错误而退出时(接收到ERR信号),输出错误发生的行号($LINENO
变量)并以状态1退出。
数组操作
array=(元素1 元素2 元素3)
${array[索引]}
${#array[@]}
字符串操作
${字符串:起始位置:长度}
${字符串/旧字符串/新字符串}
#!/bin/bash
# 数组操作
fruits=("apple" "banana" "cherry")
echo "First fruit: ${fruits[0]}"
echo "All fruits: ${fruits[@]}"
echo "Number of fruits: ${#fruits[@]}"
# 字符串操作
greeting="Hello, World!"
echo "Substring: ${greeting:7:5}"
replaced_greeting=${greeting/World/Shell}
echo "Replaced Greeting: $replaced_greeting"
数组操作部分:
fruits=("apple" "banana" "cherry")
- 这行创建了一个名为
fruits
的数组,包含三个元素:apple
、banana
和cherry
。echo "First fruit: ${fruits[0]}"
- 这行打印数组
fruits
的第一个元素(索引为0)。结果是apple
。echo "All fruits: ${fruits[@]}"
- 这行打印数组
fruits
的所有元素。${fruits[@]}
表示数组中的所有元素。echo "Number of fruits: ${#fruits[@]}"
- 这行打印数组
fruits
的长度,即它包含的元素数量。${#fruits[@]}
用于获取数组长度。字符串操作部分:
greeting="Hello, World!"
- 这行创建了一个名为
greeting
的字符串变量,内容为Hello, World!
。echo "Substring: ${greeting:7:5}"
- 这行打印字符串
greeting
从第7个字符开始的5个字符。结果是World
。replaced_greeting=${greeting/World/Shell}
- 这行将字符串
greeting
中的World
替换为Shell
,并将结果赋值给新变量replaced_greeting
。echo "Replaced Greeting: $replaced_greeting"
- 这行打印修改后的问候语。结果是
Hello, Shell!
。
高级特性
()
)。<()
和>()
将命令输出作为文件处理。自动化任务
cron
计划任务)。#!/bin/bash
# 子shell示例
(cd /tmp && echo "Current directory: $(pwd)")
# 自动化任务示例
backup_files () {
tar -czf "backup-$(date +%Y%m%d).tar.gz" $1
}
echo "Performing backup..."
backup_files /path/to/important/files
echo "Backup completed."
子Shell(Subshell)
子Shell是在一个新的Shell会话中执行命令的一种方式,它独立于当前的Shell。在子Shell中执行的命令不会影响当前Shell的环境(如变量和当前目录)。子Shell通常用于临时改变目录、设置局部变量或执行一系列命令,而不希望这些改变影响到当前的Shell环境。
自动化任务
自动化任务指的是通过脚本或程序自动执行一系列操作,而不需要人工干预。在Shell脚本中,自动化任务通常用于定期执行某些任务,如备份文件、监控系统状态、自动更新等。