[Conhecimento avançado de operação e manutenção] 34 exercícios e respostas de programação de shell (do básico ao combate real: básico + cálculo + julgamento + loop + controle e matriz + combate real avançado)

​Este artigo cobre quase a maior parte dos exercícios de gramática do Shell, usando exercícios práticos um a um para consolidar o aprendizado do Shell, sem mais delongas, vamos começar direto.

1. Noções básicas

Exercício 1: Gere o arquivo "2018-05-22.log" de acordo com a hora e escreva o status de uso diário do disco no arquivo correspondente à data

[root@Shell test]# cat disk_status_backup.sh
#!/bin/bash
name=`date +%F`
/bin/df -h > /tmp/"$name".log
[root@Shell test]# sh disk_status_backup.sh
[root@Shell test]# cat /tmp/2023-05-17.log 
Filesystem      Size  Used Avail Use% Mounted on
devtmpfs        476M     0  476M   0% /dev
tmpfs           487M     0  487M   0% /dev/shm
tmpfs           487M   14M  473M   3% /run
tmpfs           487M     0  487M   0% /sys/fs/cgroup
/dev/sda3        19G  2.0G   17G  11% /
/dev/sda1       197M  110M   88M  56% /boot
tmpfs            98M     0   98M   0% /run/user/0
[root@Shell test]# crontab -l
59 23 * * * sh /server/scripts/test/disk_status_backup.sh

Exercício 2: Conte o número de visitas de cada IP no log do Nginx. O formato do log é o seguinte

192.168.56.1 - - [21/May/2018:20:44:06 -0400] "GET /index.html HTTP/1.0" 404 169 "-" "ApacheBench/2.3" "-"/code/index.html

[root@Shell test]# cat count_nginx_ip.sh
#!/bin/bash
cat /var/log/nginx/access.log-20230506|awk '{print $1}'|sort -rn|uniq -c
[root@Shell test]# sh count_nginx_ip.sh
    207 172.16.1.5

Exercício 3: Escreva um script para calcular o tamanho da memória ocupada por todos os processos no sistema Linux e a
coluna %MEM é a porcentagem de memória física ocupada pelo processo. Você pode ver a porcentagem de memória física usada pelo processo observando esse valor.
VSZ (Virtual Memory Size) é o tamanho da memória virtual, indicando o tamanho do espaço de memória virtual utilizado pelo processo.
RSS (Resident Set Size) é o tamanho da memória física realmente utilizada pelo processo, ou seja, a quantidade de memória ocupada, em KB.
Encontramos a soma de RSS

[root@Shell test]# cat process_occupies_memory_size.sh
#!/bin/bash
ps axu|awk 'NR!=1{print $6}'|grep -v ^0$|awk '{sum+= $1}END{print sum}'
[root@Shell test]# sh process_occupies_memory_size.sh
2038676

Exercício 4: Crie 10 arquivos .txt em /backup, localize todos os arquivos com sufixos .txt no diretório /backup,
modifique txt para txt.bak em lotes e empacote e comprima todos os arquivos .bak em 123.tar.gz, o nome do arquivo de restauração em lote, exclua o .bak adicionado

[root@Shell test]# cat file_handle.sh
#!/bin/bash
for i in `ls /backup`
do
	cp /backup/"$i" /backup/"$i".bak
done
tar zcvf /backup/123.tar.gz /backup/*.txt.bak &> /dev/null
rm -rf /backup/*.txt.bak
[root@Shell test]# sh file_handle.sh
[root@Shell test]# ls /backup/
10.txt      1.txt  3.txt  5.txt  7.txt  9.txt
123.tar.gz  2.txt  4.txt  6.txt  8.txt
[root@Shell test]# tar tf /backup/123.tar.gz 
backup/10.txt.bak
backup/1.txt.bak
backup/2.txt.bak
backup/3.txt.bak
backup/4.txt.bak
backup/5.txt.bak
backup/6.txt.bak
backup/7.txt.bak
backup/8.txt.bak
backup/9.txt.bak

2. Cálculo

Exercício 1: Insira dois inteiros e calcule o resultado da adição, subtração, multiplicação, divisão e resto desses dois inteiros.

[root@Shell test]# cat jisuan.sh
#/bin/bash
num1=$1
num2=$2

echo "两数相加为:$num1 + $num2 = $(awk "BEGIN{print $num1+$num2}")"
echo "两数相减为:$num1 - $num2 = $(awk "BEGIN{print $num1-$num2}")"
echo "两数相乘为:$num1 * $num2 = $(awk "BEGIN{print $num1*$num2}")"
echo "两数相除为:$num1 / $num2 = $(awk "BEGIN{print $num1/$num2}")"
echo "两数求余为:$num1 % $num2 = $(awk "BEGIN{print $num1%$num2}")"
[root@Shell test]# sh jisuan.sh 9 3
两数相加为:9 + 3 = 12
两数相减为:9 - 3 = 6
两数相乘为:9 * 3 = 27
两数相除为:9 / 3 = 3
两数求余为:9 % 3 = 0

Exercício 2: Excluir linhas contendo letras nas primeiras cinco linhas de um documento de texto e excluir todas as letras nas linhas 6 a 10

[root@Shell test]# cat delete.sh
#!/bin/bash

# 指定待处理的文本文档路径
file_path="your_file_path.txt"

# 临时文件路径
temp_file="temp.txt"

# 删除前五行中包含字母的行
sed '1,5 { /[a-zA-Z]/d }' "$file_path" > "$temp_file"

# 删除6到10行中的所有字母
sed '6,10 s/[a-zA-Z]//g' "$temp_file" > "$file_path"

# 删除临时文件
rm "$temp_file"

Exercício 3: Imprima as palavras com menos de 3 letras na seguinte frase, eu sou koten tenho 18 anos

[root@Shell test]# cat print.sh
#!/bin/bash

sentence="I am koten I am 18"

echo "$sentence" | awk '{ for(i=1; i<=NF; i++) { if (length($i) < 3) print $i } }'

Exercício 4: Escreva um shell para ver se existem usuários personalizados (usuários comuns) no sistema linux.Se sim, quantos existem?

[root@Shell test]# cat user.sh 
#!/bin/bash

# 获取系统中普通用户的数量
user_count=$(awk -F: '$3>=1000 && $1!="nobody" {print $1}' /etc/passwd | wc -l)

if [ $user_count -gt 0 ]; then
    echo "系统中存在 $user_count 个自定义普通用户。"
else
    echo "系统中没有自定义普通用户。"
fi

Exercício 5: Escreva um script Shell para ver quais comandos você mais usa, liste seus comandos mais usados ​​top10

[root@Shell test]# cat history_cmd.sh
#!/bin/bash

# 指定要统计的shell历史文件路径
history_file=~/.bash_history

# 统计命令使用频率,并获取前10个最常用的命令
top_commands=$(cat "$history_file" | awk '{print $1}' | sort | uniq -c | sort -nr | head -n 10)

echo "您使用最多的命令 Top 10:"
echo "$top_commands"

Exercício 6: Escreva um script para calcular a soma de todos os números divisíveis por 3 dentro de 100

[root@Shell test]# cat 3.sh 
#!/bin/bash

sum=0

# 循环遍历1到100之间的数字
for ((num=1; num<=100; num++))
do
    # 判断数字是否能被3整除
    if [ $((num % 3)) -eq 0 ]; then
        sum=$((sum + num))
    fi
done

echo "100以内所有能被3整除的数字的和为:$sum"

3. Julgamento

Exercício 1: Escreva a seguinte expressão condicional como uma instrução condicional if
[ -f /etc/hosts ] && echo !

[root@Shell test]# cat if.sh 
#!/bin/bash

if [ -f /etc/hosts ]; then
    echo "!"
fi

Exercício 2: Escreva um script para realizar o julgamento dos IPs dos usuários online atuais na rede 10.0.0.0/24

[root@Shell test]# cat ping.sh
#!/bin/bash

network="10.0.0"

# 循环遍历IP地址范围内的所有地址
for ((i=1; i<=255; i++))
do
    ip="${network}.${i}"
    
    # 使用ping命令检查IP地址是否在线
    ping -c 1 -w 1 "$ip" > /dev/null 2>&1
    
    # 检查ping命令的返回值,如果为0则表示IP地址在线
    if [ $? -eq 0 ]; then
        echo "IP地址 $ip 在线"
    fi
done

Exercício 3: Use o shell para processar o seguinte conteúdo, classifique por frequência de palavras em ordem decrescente e frequência de letras em ordem decrescente.
o projeto squid fornece vários recursos para ajudar os usuários a projetar, implementar e dar suporte a instalações squid. navegue pela documentação e seções de suporte para obter mais informações

[root@Shell test]# cat sort.sh
#!/bin/bash

content="the squid project provides a number of resources to assist users design, implement and support squid installations. Please browse the documentation and support sections for more information."

# 按单词出现频率降序排序
word_frequency=$(echo "$content" | awk '{ for(i=1; i<=NF; i++) { count[$i]++ } } END { for(word in count) { print count[word], word } }' | sort -rn)

echo "按单词出现频率降序排序:"
echo "$word_frequency"

echo

# 按字母出现频率降序排序
letter_frequency=$(echo "$content" | grep -o . | grep -v '[. ]' | sort | uniq -c | sort -rn)

echo "按字母出现频率降序排序:"
echo "$letter_frequency"

Exercício 4: Use ps aux para visualizar o processo do sistema e descobrir que 100 scripts test.sh estão em execução, como eliminar todos os test.sh

[root@Shell test]# cat kill.sh
ps aux | grep test.sh | awk '{print $2}' | xargs kill

Exercício 5: Escreva um script de adivinhação de números. Quando o número digitado pelo usuário for o mesmo que o número predefinido (um número de 0 a 100 é gerado aleatoriamente), saia, caso contrário, deixe o usuário continuar inserindo e solicite ao usuário que o número é maior que o número predefinido ou Pequeno

Clique neste artigo para ver o código

Exercício 6: Use o Shell para imprimir as linhas com apenas um número em um documento de texto

[root@Shell test]# cat print_num.sh
#!/bin/bash

filename="your_text_file.txt"

while read -r line; do
  # 使用正则表达式匹配只有一个数字的行
  if [[ $line =\~ ^[0-9]+$ ]]; then
    echo "$line"
  fi
done < "$filename"

Exercício 7: Escreva um shell script para determinar se o site visitado é normal por meio do código de status retornado pelo comando curl. Por exemplo, o código de status atual é 200, que é considerado normal

[root@Shell test]# cat curl.sh
#!/bin/bash

# 设置要访问的网站URL
website="https://www.baidu.com"

# 发送GET请求并获取状态码
status_code=$(curl -s -o /dev/null -w "%{http_code}" "$website")

# 判断状态码是否为200
if [ "$status_code" -eq 200 ]; then
  echo "Website is accessible!"
else
  echo "Website is not accessible! Status code: $status_code"
fi

4. Ciclo

Exercício 1: Usando um loop while para adicionar 1 a 100

[root@Shell test]# cat add.sh
#!/bin/bash

sum=0
counter=1

while [ $counter -le 100 ]
do
    sum=$((sum + counter))
    counter=$((counter + 1))
done

echo "1加到100的和为: $sum"

Exercício 2: verifique o disco, o status de uso atual e envie o resultado para /var/log/disk.err se a taxa de uso exceder 80%

[root@Shell test]# cat disk.sh
#!/bin/bash

# 设置磁盘使用率阈值
threshold=80

# 执行df命令获取磁盘使用状态,并使用awk过滤出使用率超过阈值的行
df_output=$(df -h | awk '$5 > threshold')

# 检查是否有超过阈值的行
if [ -n "$df_output" ]; then
  echo "$df_output" >> /var/log/disk.err
  echo "Disk usage exceeded the threshold and logged to /var/log/disk.err"
else
  echo "Disk usage is within the threshold"
fi

Exercício 3: O script cria 10 usuários em lotes e a senha é uniformemente 123. É necessário julgar se a entrada do usuário é um número e se o nome da entrada está vazio

[root@Shell test]# cat useradd.sh
#!/bin/bash

count=1

while [ $count -le 10 ]
do
    echo "请输入第 $count 个用户的名字:"
    read username
    
    # 判断用户名是否为空
    if [ -z "$username" ]; then
        echo "用户名不能为空,请重新输入!"
        continue
    fi
    
    echo "请输入第 $count 个用户的密码:"
    read password
    
    # 判断密码是否为数字
    if ! [[ $password =\~ ^[0-9]+$ ]]; then
        echo "密码必须为数字,请重新输入!"
        continue
    fi
    
    # 创建用户并设置密码
    sudo useradd -m $username
    echo "$username:$password" | sudo chpasswd
    
    count=$((count + 1))
done

Exercício 4: Escreva scripts de inicialização nginx usando instruções case

Clique neste artigo para ver o código

Exercício 5: Programa de loteria. Depois de executar o script, os alunos que desejam ir inserem seus nomes completos em inglês e geram números aleatórios entre 01 e 99. Quanto maior o número, maior a probabilidade de participarem da prática do projeto. Os números que foram capturados antes não podem ser usados ​​na próxima vez O mesmo número aparece novamente; depois que o primeiro aluno digita o nome, a tela exibe informações e registra o nome e o número no arquivo. O programa não pode sair e continua esperando que outros alunos entrada.

Clique neste artigo para ver o código

Exercício 6: Imprima um menu como a seguir e, em seguida, use o loop mais a instrução case para exibir os resultados das opções de menu inseridas pelo usuário
h display command help
f exibir informações de login
d exibir montagens de disco
m verificar o uso da memória
u verificar a carga do sistema
q sair do programa

Clique neste artigo para ver o código

Cinco, controle e matriz

Exercício 1: Gere números aleatórios entre 0-100 e some-os até que sejam maiores que 1000, e mostre o resultado da adição

[root@Shell test]# cat addnum.sh
#!/bin/bash

sum=0

while [ $sum -le 1000 ]
do
    # 生成0-100之间的随机数
    random_number=$((RANDOM % 101))
    sum=$((sum + random_number))
done

echo "和大于1000的结果为: $sum"

Exercício 2: Gerar números aleatórios entre 0-100 e somá-los até que sejam maiores que 1000 e julgar se o último número aleatório é divisível por 3

[root@Shell test]# cat 3.sh
#!/bin/bash

sum=0
last_random_number=0

while [ $sum -le 1000 ]
do
    # 生成0-100之间的随机数
    random_number=$((RANDOM % 101))
    sum=$((sum + random_number))
    last_random_number=$random_number
done

echo "和大于1000的结果为: $sum"

# 判断最后一个随机数字是否能被3整除
if [ $((last_random_number % 3)) -eq 0 ]; then
    echo "最后一个随机数字 $last_random_number 能被3整除"
else
    echo "最后一个随机数字 $last_random_number 不能被3整除"
fi

Exercício 3: Determine se há um arquivo maior que 4K no diretório /tmp/ e, em caso afirmativo, imprima o tamanho e a hora de criação do arquivo

[root@Shell test]# cat 4K.sh
#!/bin/bash

path="/tmp/"

# 使用find命令查找/tmp/目录下大于4K的文件,并使用-printf选项输出文件大小和创建时间
find "$path" -type f -size +4k -printf "文件:%p,大小:%s字节,创建时间:%TY-%Tm-%Td %TH:%TM:%TS\n"

Exercício 4: Array array=(1 2 3 4 5 6) use o script para imprimir cada elemento, exibindo um elemento por linha

[root@Shell test]# cat array.sh
#!/bin/bash

array=(1 2 3 4 5 6)

# 使用for循环遍历数组
for element in "${array[@]}"
do
    echo "$element"
done

Exercício 5: Use um array para determinar as palavras com mais de 6 letras em I am koten welcome to my hometown

[root@Shell test]# cat array_6.sh
#!/bin/bash

sentence="I am koten welcome to my hometown"
words=($sentence)

# 使用for循环遍历数组中的每个单词
for word in "${words[@]}"
do
    # 统计单词中的字母数
    length=${#word}

    # 判断字母数是否大于6
    if [ $length -gt 6 ]; then
        echo "$word"
    fi
done

6. Avanço no combate real

Exercício 1: Detectar se arquivos importantes no servidor foram modificados e relatar à polícia se modificados (usando impressões digitais)

[root@Shell test]# cat fingerprint.sh
#!/bin/bash

# 定义重要文件的路径和对应的指纹
declare -A important_files=(
    ["/path/to/file1"]="fingerprint1"
    ["/path/to/file2"]="fingerprint2"
    ["/path/to/file3"]="fingerprint3"
)

# 检测文件是否被修改
for file in "${!important_files[@]}"; do
    # 计算当前文件的指纹
    current_fingerprint=$(md5sum "$file" | awk '{print $1}')

    # 比较当前指纹和预设的指纹
    if [[ "$current_fingerprint" != "${important_files[$file]}" ]]; then
        echo "警报!文件 $file 已被修改!"
        # 在此处添加触发报警的操作,比如发送邮件或短信通知管理员
    fi
done

Exercício 2: Calcule a síntese de tráfego usada pelos 10 IPs mais acessados ​​no log nginx

[root@Shell test]# cat ip.sh
#!/bin/bash

logfile="access.log"

# 使用awk命令提取日志中的IP和流量信息,并按照IP统计流量总和
ip_traffic=$(awk '{ ip[$1] += $10 } END { for(i in ip) print i, ip[i] }' "$logfile")

# 使用sort命令对流量进行降序排序
sorted_ip_traffic=$(echo "$ip_traffic" | sort -k2 -rn)

# 使用head命令获取前10个IP和对应的流量
top_10_ip_traffic=$(echo "$sorted_ip_traffic" | head -n 10)

# 输出结果
echo "访问最多的10个IP使用的流量综合:"
echo "$top_10_ip_traffic"

Exercício 3: Evite ataques do DOS, verifique os logs do nginx e, se o PV de um determinado IP for muito grande por um curto período de tempo, use o firewall para bani-lo

[root@Shell test]# cat iptables.sh
#!/bin/bash

logfile="access.log"
threshold=100  # 设置阈值,表示短时间内的页面访问量上限
ban_time=60    # 设置禁用时间,单位为秒

# 使用awk命令提取日志中的IP信息,并统计每个IP的页面访问量
ip_pv=$(awk '{ ip[$1]++ } END { for(i in ip) print i, ip[i] }' "$logfile")

# 使用while循环遍历PV统计结果
while IFS= read -r line; do
    ip=$(echo "$line" | awk '{print $1}')
    pv=$(echo "$line" | awk '{print $2}')

    # 判断页面访问量是否超过阈值
    if [ "$pv" -gt "$threshold" ]; then
        echo "警报!IP $ip 的短时间内页面访问量过大!"

        # 使用防火墙命令禁用IP
        iptables -A INPUT -s "$ip" -j DROP

        # 等待一段时间后解禁IP
        sleep "$ban_time"
        iptables -D INPUT -s "$ip" -j DROP
    fi
done <<< "$ip_pv"

Exercício 4: O uso da CPU do servidor de monitoramento em tempo real é superior a 80%, o uso do disco é superior a 80% e o uso da memória é superior a 80% e alarmes

[root@Shell test]# cat cpu_disk.sh
#!/bin/bash

# 配置监控阈值
cpu_threshold=80
disk_threshold=80
memory_threshold=80

# 检查CPU使用率
check_cpu() {
    cpu_usage=$(sar -u 1 1 | grep Average | awk '{print 100 - $NF}')

    if [ "$cpu_usage" -gt "$cpu_threshold" ]; then
        echo "警报!CPU使用率超过 $cpu_threshold%!"
        # 在此处添加触发报警的操作,比如发送邮件通知管理员
    fi
}

# 检查磁盘使用率
check_disk() {
    disk_usage=$(df -h | awk '/\/$/ {print $5}' | sed 's/%//')

    if [ "$disk_usage" -gt "$disk_threshold" ]; then
        echo "警报!磁盘使用率超过 $disk_threshold%!"
        # 在此处添加触发报警的操作,比如发送邮件通知管理员
    fi
}

# 检查内存使用率
check_memory() {
    memory_usage=$(free | awk '/Mem/ {printf "%.2f", $3/$2 * 100}')

    if (( $(echo "$memory_usage > $memory_threshold" | bc -l) )); then
        echo "警报!内存使用率超过 $memory_threshold%!"
        # 在此处添加触发报警的操作,比如发送邮件通知管理员
    fi
}

# 持续监控
while true; do
    check_cpu
    check_disk
    check_memory
    sleep 5
done

Exercício 5: Instale o nginx usando binário, exigindo que os scripts sejam reexecutáveis

[root@Shell test]# cat nginx.sh
#!/bin/bash

nginx_version="1.18.0"  # 设置nginx版本号
nginx_install_dir="/usr/local/nginx"  # 设置nginx安装目录

# 下载nginx源码包
wget http://nginx.org/download/nginx-$nginx_version.tar.gz

# 解压源码包
tar -zxvf nginx-$nginx_version.tar.gz

# 进入解压后的目录
cd nginx-$nginx_version

# 配置编译参数
./configure --prefix=$nginx_install_dir

# 编译和安装
make && make install

# 清理临时文件
cd ..
rm -rf nginx-$nginx_version
rm nginx-$nginx_version.tar.gz

Exercício 6: Faça backup do sub-banco de dados e da subtabela do banco de dados para /tmp.

[root@Shell test]# cat mysql.sh
#!/bin/bash

# 定义数据库连接信息
host=localhost
port=3306
user=root
password=123456

# 定义要备份的数据库和表
db_list=(db1 db2)
table_list=(table1 table2 table3)

# 遍历数据库
for db in ${db_list[@]}; do
  # 遍历表
  for table in ${table_list[@]}; do
    # 构建备份文件名
    bak_file=/tmp/${db}_${table}.sql
    
    # 执行备份命令
    mysqldump -h$host -P$port -u$user -p$password $db $table > $bak_file
    
    # 检查备份是否成功
    if [ $? -eq 0 ]; then
      echo "备份 $db.$table 成功"
    else
      echo "备份 $db.$table 失败"
    fi
  done
done

Sou koten, com 10 anos de experiência em operação e manutenção, e continuo compartilhando secos e secos em operação e manutenção, obrigado pela leitura e atenção!

Acho que você gosta

Origin blog.csdn.net/qq_37510195/article/details/131631010
Recomendado
Clasificación